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开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

【QT】基础入门学习

文章目录 浅析Qt应用程序的主函数使用qDebug()函数常用快捷键Qt 编码风格信号槽连接模型实现方案 信号和槽的工作机制Qt对象树机制 浅析Qt应用程序的主函数 #include "mywindow.h"#include <QApplication>// 程序的入口int main(int argc, char *argv[]){// argc是命令行参数个数,argv是

Python QT实现A-star寻路算法

目录 1、界面使用方法 2、注意事项 3、补充说明 用Qt5搭建一个图形化测试寻路算法的测试环境。 1、界面使用方法 设定起点: 鼠标左键双击,设定红色的起点。左键双击设定起点,用红色标记。 设定终点: 鼠标右键双击,设定蓝色的终点。右键双击设定终点,用蓝色标记。 设置障碍点: 鼠标左键或者右键按着不放,拖动可以设置黑色的障碍点。按住左键或右键并拖动,设置一系列黑色障碍点

使用Qt编程QtNetwork无法使用

使用 VS 构建 Qt 项目时 QtNetwork 无法使用的问题 - 摘叶飞镖 - 博客园 (cnblogs.com) 另外,强烈建议在使用QNetworkAccessManager之前看看这篇文章: Qt 之 QNetworkAccessManager踏坑记录-CSDN博客 C++ Qt开发:QNetworkAccessManager网络接口组件 阅读目录 1.1 通用API函数

Qt多语种开发教程

Qt作为跨平台的开发工具,早已应用到各行各业的软件开发中。 今天讲讲,Qt开发的正序怎么做多语言开发。就是说,你设置中文,就中文显示;设置英语就英文显示,设置繁体就繁体显示,设置发育就显示法语等。 开发环境(其实多语种这块根环境没太大关系):win10,Qt.5.12.10 一.先用QtCreator创建一个简单的桌面程序 1.工程就随便命名“LanguageTest”,其他默认。 2.在设计师

Weex入门教程之4,获取当前全局环境变量和配置信息(屏幕高度、宽度等)

$getConfig() 获取当前全局环境变量和配置信息。 Returns: config (object): 配置对象;bundleUrl (string): bundle 的 url;debug (boolean): 是否是调试模式;env (object): 环境对象; weexVersion (string): Weex sdk 版本;appName (string): 应用名字;

Weex入门教程之3,使用 Vue 开发 Weex 页面

环境安装 在这里简略地介绍下,详细看官方教程 Node.js 环境 Node.js官网 通常,安装了 Node.js 环境,npm 包管理工具也随之安装了。因此,直接使用 npm 来安装 weex-toolkit。 npm 是一个 JavaScript 包管理工具,它可以让开发者轻松共享和重用代码。Weex 很多依赖来自社区,同样,Weex 也将很多工具发布到社区方便开发者使用。

Weex入门教程之2,Android Studio安装Weex插件

插件位置及描述 https://plugins.jetbrains.com/idea/plugin/8460-weex 貌似对windows还不是很支持,先放着吧。 安装 插件功能 先预览下都有什么功能 安装完成Weex插件后,如果在main toolbar找不到这些功能图标,那么就需要手动添加到main toolbar 添加到main toolbar 红框内就是

Weex入门教程之1,了解Weex

【资料合集】Weex Conf回顾集锦:讲义PDF+活动视频! PDF分享:链接:http://pan.baidu.com/s/1hr8RniG 密码:fa3j 官方教程:https://weex-project.io/cn/v-0.10/guide/index.html 用意 主要是介绍Weex,并未涉及开发方面,好让我们开始开发之前充分地了解Weex到底是个什么。 以下描述主要摘取于

Qt中window frame的影响

window frame 在创建图形化界面的时候,会创建窗口主体,上面会多出一条,周围多次一圈细边,这就叫window frame窗口框架,这是操作系统自带的。 这个对geometry的一些属性有一定影响,主要体现在Qt坐标系体系: 窗口当中包含一个按钮,这个按钮的坐标系是以父元素为参考,那么这个参考是widget本体作为参考,还是window frame作为参考,这两种参考体系都存在