Qt+OpenGL入门教程(四)——VBO、VAO和EBO

2024-04-02 06:12
文章标签 qt 入门教程 opengl vao vbo ebo

本文主要是介绍Qt+OpenGL入门教程(四)——VBO、VAO和EBO,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前面我们已经简单绘制了一个三角形,但这只是个小demo是远远不够的,当顶点数据很多时,解析很麻烦时我们应该如何处理呢?接下来我们来介绍一下在OpenGL开发中帮助我们提升渲染性能的几种数据对象。

注意:所有代码都是基于上一篇修改,看代码时候,一定多看看看我写的注释!

VBO(Vertex Buffer Object)顶点缓冲对象

VBO会在GPU上创建内存,用于存储我们的顶点数据。
VBO的缓冲类型是GL_ARRAY_BUFFER。

代码如下:

widget.cpp

#include "widget.h"GLuint VBO;  // VBOWidget::Widget(QWidget *parent): QOpenGLWidget(parent)
{
}Widget::~Widget()
{
}void Widget::initializeGL()
{// 初始化OpenGL函数,将Qt里面的函数指针指向显卡的函数initializeOpenGLFunctions();glClearColor(0.0f, 0.0f, 0.0f, 1.0f);glClear(GL_COLOR_BUFFER_BIT);shaderProgram.create();shaderProgram.addShaderFromSourceFile(QOpenGLShader::Vertex, ":/gl.vert");shaderProgram.addShaderFromSourceFile(QOpenGLShader::Fragment, ":/gl.frag");shaderProgram.link();GLfloat vertices[] = {0.0f, 0.5f, 0.0f,0.5f, -0.5f, 0.0f,-0.5f, -0.5f, 0.0f,};// 创建VBOglGenBuffers(1, &VBO);// 绑定VBOglBindBuffer(GL_ARRAY_BUFFER, VBO);// 为当前绑定到target的缓冲区对象创建一个新的数据存储glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices, GL_STATIC_DRAW);// 解绑glBindBuffer(GL_ARRAY_BUFFER, 0);
}void Widget::resizeGL(int w, int h)
{glViewport(0, 0, w, h);
}void Widget::paintGL()
{shaderProgram.bind();// 启用顶点属性(允许顶点着色器读取GPU数据)glEnableVertexAttribArray(0);// 绑定顶点缓冲对象glBindBuffer(GL_ARRAY_BUFFER, VBO);// 设置解析规则,GPU能够取到正确的数据供着色器使用glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 3 * sizeof(GLfloat), (void*)0);glDrawArrays(GL_TRIANGLES, 0, 3);
}

总结:

将顶点数据存储在VBO中可以减少CPU与GPU之间的数据传输次数,提高渲染效率!

VAO(Vertex Array Object)顶点数组对象

VAO用于存储多个顶点属性的状态设置。它包含了一系列函数调用的状态,这些函数调用设置了顶点属性指针和绑定了顶点缓冲区对象(VBO),以便在绘制时使用。

代码如下:

widget.cpp

#include "widget.h"GLuint VBO;  // VBO
GLuint VAO;  // VAOWidget::Widget(QWidget *parent): QOpenGLWidget(parent)
{
}Widget::~Widget()
{
}void Widget::initializeGL()
{// 初始化OpenGL函数,将Qt里面的函数指针指向显卡的函数initializeOpenGLFunctions();glClearColor(0.0f, 0.0f, 0.0f, 1.0f);glClear(GL_COLOR_BUFFER_BIT);shaderProgram.create();shaderProgram.addShaderFromSourceFile(QOpenGLShader::Vertex, ":/gl.vert");shaderProgram.addShaderFromSourceFile(QOpenGLShader::Fragment, ":/gl.frag");shaderProgram.link();GLfloat vertices[] = {0.0f, 0.5f, 0.0f,0.5f, -0.5f, 0.0f,-0.5f, -0.5f, 0.0f,};// 创建VAOglGenVertexArrays(1, &VAO);// 创建VBOglGenBuffers(1, &VBO);// 绑定VAO(VAO是没有缓冲类型的)glBindVertexArray(VAO);// 绑定VBOglBindBuffer(GL_ARRAY_BUFFER, VBO);// 为当前绑定到target的缓冲区对象创建一个新的数据存储glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices, GL_STATIC_DRAW);// 设置顶点属性指针glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 3 * sizeof(GLfloat), (void*)0);// 启用顶点属性(允许顶点着色器读取GPU数据)glEnableVertexAttribArray(0);// 解绑glBindBuffer(GL_ARRAY_BUFFER, 0);glBindVertexArray(0);
}void Widget::resizeGL(int w, int h)
{glViewport(0, 0, w, h);
}void Widget::paintGL()
{shaderProgram.bind();glBindVertexArray(VAO);glDrawArrays(GL_TRIANGLES, 0, 3);
}

总结:

看完这个代码你会发现,paintGL函数里面缺少了顶点数据的解析,只需要绑定VAO即可。

以下是VAO的作用:

  • 简化顶点属性设置(将顶点属性设置封装在一个对象中,可以更容易地管理和使用顶点属性)
  • 提高渲染效率(通过预定义顶点属性设置状态,可以避免在每次绘制调用时重复设置相同的顶点属性)
  • 提高可读性和维护性(将顶点属性设置组织成一个单独的对象,使代码更具有可读性和维护性)**。

VAO和VBO的关系

  • VBO是纯数据的缓冲区
  • VAO是一个数组,保存每一类顶点属性的解析结果,OpenGL中貌似最多支持16种顶点属性,这里的顶点属性就是glVertexAttribPointer方法的第一个参数指定的,通常0表示顶点坐标,1表示顶点颜色
  • 使用VAO的好处是,你只需要针对VBO做一次解析,将结果存储到VAO中,每一帧渲染使用VAO的指针来访问缓冲区数据,而不需要每一帧都做解析

EBO/IBO(Element/Index Buffer Object)索引缓冲对象

既然有了VBO和VAO,那EBO还有什么用途呢?接下来我们一起研究一下!

EBO主要用来存储顶点的索引信息,那为什么需要存储索引呢?

举个栗子:

假如我们要绘制两个三角形来组成一个矩形(OpenGL主要处理三角形),顶点数据应该是这样的。

GLfloat vertices[] = {0.5f, 0.5f, 0.0f,    // 右上0.5f, -0.5f, 0.0f,   // 右下-0.5f, -0.5f, 0.0f,  // 左下-0.5f, -0.5f, 0.0f,  // 左下-0.5f, 0.5f, 0.0f,   // 左上0.5f, 0.5f, 0.0f,    // 右上
};

细心的你可能会发现有两个点的坐标是重复的,那么这样是不就相当于存储了无用数据嘛?是不是降低了传输效率呢?所以就用到了EBO。

代码如下:

widget.cpp

#include "widget.h"GLuint VBO;  // VBO
GLuint VAO;  // VAO
GLuint EBO;  // EBOWidget::Widget(QWidget *parent): QOpenGLWidget(parent)
{
}Widget::~Widget()
{
}void Widget::initializeGL()
{// 初始化OpenGL函数,将Qt里面的函数指针指向显卡的函数initializeOpenGLFunctions();glClearColor(0.0f, 0.0f, 0.0f, 1.0f);glClear(GL_COLOR_BUFFER_BIT);shaderProgram.create();shaderProgram.addShaderFromSourceFile(QOpenGLShader::Vertex, ":/gl.vert");shaderProgram.addShaderFromSourceFile(QOpenGLShader::Fragment, ":/gl.frag");shaderProgram.link();// 去重后的顶点数据GLfloat vertices[] = {0.5f, 0.5f, 0.0f,0.5f, -0.5f, 0.0f,-0.5f, -0.5f, 0.0f,-0.5f, 0.5f, 0.0f,};// 顶点下标索引GLuint indices[] = {0, 1, 2,0, 2, 3};// 创建VAOglGenVertexArrays(1, &VAO);// 创建VBOglGenBuffers(1, &VBO);// 绑定VAO(VAO是没有缓冲类型的)glBindVertexArray(VAO);// 绑定VBOglBindBuffer(GL_ARRAY_BUFFER, VBO);// 为当前绑定到target的缓冲区对象创建一个新的数据存储glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices, GL_STATIC_DRAW);// 设置顶点属性指针glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 3 * sizeof(GLfloat), (void*)0);// 启用顶点属性(允许顶点着色器读取GPU数据)glEnableVertexAttribArray(0);// 解绑VBOglBindBuffer(GL_ARRAY_BUFFER, 0);// 创建EBOglGenBuffers(1, &EBO);// 绑定EBOglBindBuffer(GL_ELEMENT_ARRAY_BUFFER, EBO);glBufferData(GL_ELEMENT_ARRAY_BUFFER, sizeof(indices), indices, GL_STATIC_DRAW);// 解绑VAO(在EBO后解绑,paintGL中就不用绑定EBO了)glBindVertexArray(0);
}void Widget::resizeGL(int w, int h)
{glViewport(0, 0, w, h);
}void Widget::paintGL()
{shaderProgram.bind();glBindVertexArray(VAO);glDrawElements(GL_TRIANGLES, 6, GL_UNSIGNED_INT, NULL);
}

总结:

  • 索引数组的类型必须用GLuint,不要误用GLfloat,否则将无法绘制成功
  • 使用EBO可以减少需要传输到GPU的数据量。相比直接传输顶点数据,使用索引数组可以更有效地管理和重用顶点数据
  • 绘制函数发生变化,不再是glDrawArrays而是glDrawElements

这篇关于Qt+OpenGL入门教程(四)——VBO、VAO和EBO的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/869191

相关文章

Qt QCustomPlot库简介(最新推荐)

《QtQCustomPlot库简介(最新推荐)》QCustomPlot是一款基于Qt的高性能C++绘图库,专为二维数据可视化设计,它具有轻量级、实时处理百万级数据和多图层支持等特点,适用于科学计算、... 目录核心特性概览核心组件解析1.绘图核心 (QCustomPlot类)2.数据容器 (QCPDataC

Qt如何实现文本编辑器光标高亮技术

《Qt如何实现文本编辑器光标高亮技术》这篇文章主要为大家详细介绍了Qt如何实现文本编辑器光标高亮技术,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以了解下... 目录实现代码函数作用概述代码详解 + 注释使用 QTextEdit 的高亮技术(重点)总结用到的关键技术点应用场景举例示例优化建议

Qt 设置软件版本信息的实现

《Qt设置软件版本信息的实现》本文介绍了Qt项目中设置版本信息的三种常用方法,包括.pro文件和version.rc配置、CMakeLists.txt与version.h.in结合,具有一定的参考... 目录在运行程序期间设置版本信息可以参考VS在 QT 中设置软件版本信息的几种方法方法一:通过 .pro

VS配置好Qt环境之后但无法打开ui界面的问题解决

《VS配置好Qt环境之后但无法打开ui界面的问题解决》本文主要介绍了VS配置好Qt环境之后但无法打开ui界面的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 目UKeLvb录找到Qt安装目录中designer.UKeLvBexe的路径找到vs中的解决方案资源

Qt之QMessageBox的具体使用

《Qt之QMessageBox的具体使用》本文介绍Qt中QMessageBox类的使用,用于弹出提示、警告、错误等模态对话框,具有一定的参考价值,感兴趣的可以了解一下... 目录1.引言2.简单介绍3.常见函数4.按钮类型(QMessage::StandardButton)5.分步骤实现弹窗6.总结1.引言

Qt中Qfile类的使用

《Qt中Qfile类的使用》很多应用程序都具备操作文件的能力,包括对文件进行写入和读取,创建和删除文件,本文主要介绍了Qt中Qfile类的使用,具有一定的参考价值,感兴趣的可以了解一下... 目录1.引言2.QFile文件操作3.演示示例3.1实验一3.2实验二【演示 QFile 读写二进制文件的过程】4.

Qt实现网络数据解析的方法总结

《Qt实现网络数据解析的方法总结》在Qt中解析网络数据通常涉及接收原始字节流,并将其转换为有意义的应用层数据,这篇文章为大家介绍了详细步骤和示例,感兴趣的小伙伴可以了解下... 目录1. 网络数据接收2. 缓冲区管理(处理粘包/拆包)3. 常见数据格式解析3.1 jsON解析3.2 XML解析3.3 自定义

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

Qt中QGroupBox控件的实现

《Qt中QGroupBox控件的实现》QGroupBox是Qt框架中一个非常有用的控件,它主要用于组织和管理一组相关的控件,本文主要介绍了Qt中QGroupBox控件的实现,具有一定的参考价值,感兴趣... 目录引言一、基本属性二、常用方法2.1 构造函数 2.2 设置标题2.3 设置复选框模式2.4 是否

QT进行CSV文件初始化与读写操作

《QT进行CSV文件初始化与读写操作》这篇文章主要为大家详细介绍了在QT环境中如何进行CSV文件的初始化、写入和读取操作,本文为大家整理了相关的操作的多种方法,希望对大家有所帮助... 目录前言一、CSV文件初始化二、CSV写入三、CSV读取四、QT 逐行读取csv文件五、Qt如何将数据保存成CSV文件前言