AB是一家?VAO与VBO

2024-04-06 03:48
文章标签 ab 一家 vao vbo

本文主要是介绍AB是一家?VAO与VBO,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原文地址:http://www.zwqxin.com/archives/opengl/vao-and-vbo-stuff.html

而且上面这个博客作者记录了很多opengl相关的知识点,值得收藏。

我想大家都已经熟悉VBO了吧。在GL3.0时代的VBO大体还是处于最重要的地位,但是与此同时也出现了不少新的用法和辅助役,其中一个就是VAO。本文大致小记一下这两者的联系,帮助大家理解一下这个角色。——ZwqXin.com

VBO?See[学一学,VBO]

本文来源于 ZwqXin (http://www.zwqxin.com/), 转载请注明
      原文地址:http://www.zwqxin.com/archives/opengl/vao-and-vbo-stuff.html

如果你也逐渐步进GL3.0开始的新标准,你大概会留意到传统的绘图方式(glVertex)已经要被废掉了,不仅如此,以最高绘制速度为标记的显示列表方式也已经被印上deprecated了,这样,在以前的文章([学一学,VBO] )中的讨论,在新标准的面前都显得没什么必要了。我想说的是,OpenGL对GPU的入口“顶点传送”——或者说,绘制方式,尽量不要再选择传统方式(glVertex)或显示列表(glCallList)甚至VA(vertex array)了。哪怕你是用的一个compatable的GL-context,哪怕顶点数据部分持续变化或者恒定不变,也得注意要尽量尽量使用VBO来组织你的数据。

另外的一点,就是尽量不要以客户端状态函数来使用VBO了。我是说——glEnableClientState/glDisableClientState,还有glVertexPointer这类函数。VBO的本意是把本地(GL客户端)的数据完全交给GPU(GL服务端)来管理,所以若非为了数据的更新,你完全可以在调用glBufferData之后选择扔弃保存在本地内存中的数据。VBO可以说只有在传输数据的时候跟本地客户端有联系,它的状态是服务端(我们的流水线)管理的,当初沿用VA的那些客户端状态函数,还有一个原因就是它们方便地与shader里面的固定attribute(gl_Position之类)建立联系【见[OpenGL/GLSL数据传递小记(2.x)]】,但是GLSL已经也不推荐使用那些attrbute了。(事实上,以上这些都是deprecated的了。)

C++代码1
  1. glBindBuffer(GL_ARRAY_BUFFER, m_nPositionVBO);  
  2. glEnableClientState(GL_VERTEX_ARRAY);  
  3. glVertexPointer(2, GL_FLOAT, 0, NULL);  
  4.   
  5. glBindBuffer(GL_ARRAY_BUFFER, m_nTexcoordVBO);  
  6. glEnableClientState(GL_TEXTURE_COORD_ARRAY);  
  7. glTexCoordPointer(2, GL_FLOAT, 0, NULL);  
  8.   
  9. glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_nIndexVBO);  
  10.   
  11. glDrawElements(GL_TRIANGLES, 6, GL_UNSIGNED_SHORT, NULL);  
  12.   
  13. glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, NULL);  
  14.   
  15. glDisableClientState(GL_TEXTURE_COORD_ARRAY);  
  16. glDisableClientState(GL_VERTEX_ARRAY);  
  17. glBindBuffer(GL_ARRAY_BUFFER, NULL);  
C++代码2
  1. glBindBuffer(GL_ARRAY_BUFFER, m_nQuadPositionVBO);  
  2. glEnableVertexAttribArray(VAT_POSITION);  
  3. glVertexAttribPointer(VAT_POSITION, 2, GL_INT, GL_FALSE, 0, NULL);  
  4.   
  5. glBindBuffer(GL_ARRAY_BUFFER, m_nQuadTexcoordVBO);  
  6. glEnableVertexAttribArray(VAT_TEXCOORD);  
  7. glVertexAttribPointer(VAT_TEXCOORD, 2, GL_INT, GL_FALSE, 0, NULL);  
  8.   
  9. glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_nQuadIndexVBO);  
  10.   
  11. glDrawElements(GL_TRIANGLES, 6, GL_UNSIGNED_SHORT, NULL);  
  12.   
  13. glDisableVertexAttribArray(VAT_POSITION);  
  14. glDisableVertexAttribArray(VAT_TEXCOORD);  
  15.   
  16. glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, NULL);  
  17. glBindBuffer(GL_ARRAY_BUFFER, NULL);  

以上两段是效果一致的VBO渲染部分的代码。尽量用第二种吧。使用第二种的前提是你使用shader来进行顶点处理,VAT_POSITION/VAT_TEXCOORD需要与Shader里代表顶点/纹理坐标的attribute变量建立联系(参考[OpenGL/GLSL数据传递小记(2.x)]),在这个GL3.0之后的时代里,这种前提也算不上什么前提就是了。我们来囫囵吞枣地猜测一下OpenGL是怎么处理VBO的数据的。

1. VBO

与其他buffer object一样,VBO归根到底是显卡存储空间里的一块缓存区(Buffer)而已,这个Buffer有它的名字(VBO的ID),OpenGL在GPU的某处记录着这个ID和对应的显存地址(或者地址偏移,类似内存)。用代码看看吧:

C++代码
  1.     //生成一个Buffer的ID,不管是什么类型的  
  2. glGenBuffers(1, &m_nQuadVBO);   
  3. //绑定ID,同时也指定该ID对应的buffer的信息类型是GL_ARRAY_BUFFER  
  4. glBindBuffer(GL_ARRAY_BUFFER, m_nQuadVBO);  
  5. //为该ID指定一块指定大小的存储区域(区域的位置大抵由末参数影响),  传输数据      
  6.     glBufferData(GL_ARRAY_BUFFER, sizeof(fQuadData), fQuadData, GL_STREAM_DRAW);  

这里是VBO的初始化阶段。在这里我们看到了这是对位置,还是颜色,还是纹理坐标,还是法线,还是其他顶点属性进行设置的吗?是的,这个信息是:起码在初始化阶段,一个VBO对于交给它存储的数据到底是什么,完全不知道。我们此时再看回上面两段渲染部分的代码,就明白了:哦,原来这都是在渲染时确定的!

对于第一段渲染代码,glVertexPointer(2, GL_FLOAT, 0, NULL)这个函数指定了VBO里的是什么数据——顶点位置,float类型,2个float指涉一个顶点位置,在区域里无偏移地采集数据,等等。之后的glDrawElements只不过根据组织模式(GL_TRIANGLES,这个是直接交给vertex处理后的Geometry处理的)和索引数据去采集VBO里的这些数据罢了——它从某个地方获取了glBindBuffer指定的位置,还有glVertexPointer设定的信息(由glEnableClientState启用),它进行绘制所需要的一切——这个地方,就是所谓的GL-Context吧,那个保存了所有运行时流水线状态的东西。

对于第二段渲染代码,大体是一样的,只是glVertexAttribPointer使用第一个参数(location)指涉对应vertex-shader里哪个in attribute。VBO在渲染阶段才指定数据位置和“顶点信息”(Vertex Specification),然后根据此信息去解析缓存区里的数据,联系这两者中间的桥梁是GL-Contenxt。GL-context整个程序一般只有一个,所以如果一个渲染流程里有两份不同的绘制代码,GL-context就负责在它们之间进行状态切换。这也是为什么要在渲染过程中,在每份绘制代码之中有glBindBuffer/glEnableVertexAttribArray/glVertexAttribPointer。那么优化方法就来了——把这些都放到初始化时候完成吧!——这样做的限制条件是“负责记录状态的GL-context整个程序一般只有一个”,那么就不直接用GL-context记录,用别的东西做状态记录吧——这个东西针对"每份绘制代码“有一个,记录该次绘制所需要的所有VBO所需信息,把它保存到GPU特定位置,绘制的时候直接在这个位置取信息绘制。

于是,VAO诞生了。

2.VAO

VAO的全名是Vertex Array Object,首先,它不是Buffer-Object,所以不用作存储数据;其次,它针对”顶点“而言,也就是说它跟”顶点的绘制“息息相关,在GL3.0的世界观里,这相当于”与VBO息息相关“。(提示,它跟VA真是虾米关系都没有的,嘛,虽然这的确让人误会,我最初见到这个名词时也误会了的说。)

按上所述,它的定位是state-object(状态对象,记录存储状态信息)。这明显区别于buffer-object。如果有人碎碎念”既然是记录顶点的信息,为什么不叫vertex attribute object“呢?我想说这些孩子你们真没认真看文章嘛——VAO记录的是一次绘制中做需要的信息,这包括”数据在哪里-glBindBuffer(GL_ARRAY_BUFFER)“、”数据的格式是怎样的-glVertexAttribPointer“(顶点位置的数据在哪里,顶点位置的数据的格式是怎样的/纹理坐标的数据在哪里,纹理坐标的数据的格式是怎样的....视乎你让它关联多少个VBO、VBO里有多少种数据),顺带一提的是,这里的状态还包括这些属性关联的shader-attribute的location的启用(glEnableVertexAttribArray)、这些顶点属性对应的顶点索引数据的位置(glBindBuffer(GL_ELEMENT_ARRAY_BUFFER),如果你指定了的话)。在GL的wiki里把这些”信息“抽象成一个属性数据体:

  1. struct VertexAttribute  
  2. {  
  3.     bool bIsEnabled = GL_FALSE;  
  4.     int iSize = 4; //This is the number of elements in this attribute, 1-4.  
  5.     unsigned int iStride = 0;  
  6.     VertexAttribType eType = GL_FLOAT;  
  7.     bool bIsNormalized = GL_FALSE;  
  8.     bool bIsIntegral = GL_FALSE;  
  9.     void * pBufferObjectOffset = 0;  
  10.     BufferObject * pBufferObj = 0;  
  11. };  
  12.    
  13. struct VertexArrayObject  
  14. {  
  15.     BufferObject *pElementArrayBufferObject = NULL;  
  16.     VertexAttribute attributes[GL_MAX_VERTEX_ATTRIB];  
  17. }  

这里,VertexArrayObject 就包括了一个Index-VBO【[索引顶点的VBO与多重纹理下的VBO]】(可以没有,例如绘制用的是glDrawArray)还有一些VertexAttribute。后者包括顶点属性的格式和位置和一个启用与否的状态。这些都对应了上述讨论的那几个函数(注意glVertexAttribPointer和glVertexAttribIPointer的选择决定bool bIsIntegral,数据是否整型不可规范化)。那么,现在我们可以知道VAO的用法了:

C++代码 - 初始化部分
  1. glGenVertexArrays(1, &m_nQuadVAO);  
  2. glBindVertexArray(m_nQuadVAO);  
  3.   
  4.   
  5. glGenBuffers(1, &m_nQuadPositionVBO);  
  6. glBindBuffer(GL_ARRAY_BUFFER, m_nQuadPositionVBO);  
  7. glBufferData(GL_ARRAY_BUFFER, sizeof(fQuadPos), fQuadPos, GL_STREAM_DRAW);  
  8.   
  9. glEnableVertexAttribArray(VAT_POSITION);  
  10. glVertexAttribPointer(VAT_POSITION, 2, GL_INT, GL_FALSE, 0, NULL);
  11.   
  12. glGenBuffers(1, &m_nQuadTexcoordVBO);  
  13. glBindBuffer(GL_ARRAY_BUFFER, m_nQuadTexcoordVBO);  
  14. glBufferData(GL_ARRAY_BUFFER, sizeof(fQuadTexcoord), fQuadTexcoord, GL_STREAM_DRAW);  
  15.   
  16. glEnableVertexAttribArray(VAT_TEXCOORD);  
  17. glVertexAttribPointer(VAT_TEXCOORD, 2, GL_INT, GL_FALSE, 0, NULL);  
  18.   
  19. glGenBuffers(1, &m_nQuadIndexVBO);  
  20. glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_nQuadIndexVBO);  
  21. glBufferData(GL_ELEMENT_ARRAY_BUFFER, sizeof(nQuadIndex), nQuadIndex, GL_STREAM_DRAW);  
  22.   
  23.   
  24. glBindVertexArray(NULL);  
  25.   
  26. glBindBuffer(GL_ARRAY_BUFFER, NULL);  
  27. glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, NULL);  
C++代码 - 渲染部分
  1. glBindVertexArray(m_nQuadVAO);  
  2.   
  3. glDrawElements(GL_TRIANGLES, 6, GL_UNSIGNED_SHORT, NULL);  
  4.   
  5. glBindVertexArray(NULL);  

以上就是VAO的使用方法了,很直观吧?

使用VAO的好处?看上面那么简洁的渲染部分代码就够了。

你甚至可以认为VAO就是一个状态容器,其中粗体字的那几行就是它以及它所”包含“的东西——填充了”VertexArrayObject结构体“的东西。注意:1.没有一个合适的地方给glDisableVertexAttribArray了,事实上调用glBindVertexArray(NULL)的时候里面所有状态都”关掉“了,也就没所谓针对顶点属性的location做其他什么;2.glBindBuffer(GL_ARRAY_BUFFER, NULL)/glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, NULL)一定要在glBindVertexArray(NULL)后面(不然VAO就把它们也包含了,最后就渲染不出东西了);3.glDrawElements里面的东西(顶点索引的属性状态)VAO可没记录保存哦;4.glVertexPointer那类函数理论上也可以,但是建议还是不要混用deprecated的函数进去了。

那么,既然AB是一家,那就两个站都上一上吧!哦,不对,两个O都用一用吧!

本文来源于 ZwqXin (http://www.zwqxin.com/), 转载请注明
      原文地址:http://www.zwqxin.com/archives/opengl/vao-and-vbo-stuff.html



这篇关于AB是一家?VAO与VBO的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

性能测试工具 wrk,ab,locust,Jmeter 压测结果比较

前言 在开发服务端软件时,经常需要进行性能测试,一般我采用手写性能测试代码的方式进行测试,那有什么现成的好的性能测试工具吗? 性能测试工具 wrk,ab,locust,Jmeter 压测结果比较 详见: 性能测试工具 wrk,ab,locust,Jmeter 压测结果比较 Jmeter性能测试 入门

八大建筑央企实力排行-你打算进哪一家?

第八名:中国化学 营收:1584.37亿元 中国化学业务覆盖建筑工程、环境治理、工艺工程技术开发等服务。通过持续创新和精细化管理,积极推进多个工程领域的全过程服务和产业运营、资本运营,实现公司的持续发展。 第七名:中国能建 营收:3663.93亿元 中国能源建设集团是一家为中国乃至全球能源电力、基础设施等行业提供发展方案和服务的综合性特大型集团公司,连续9年进入世界500强,业务遍布世界140多个

Apache ab 压力测试出现 Failed requests

如图:  Failed requests 分別有 Connect, Length,Exception : 1:Connect 无法送出要求,目标主机连接失败,要求的过程中连接被中断

福建聚鼎:装饰画做起来一家店铺需要多久

在如今快节奏、高效率的社会环境中,许多人追求即时满足,希望所有事情都能迅速完成。然而,在艺术的世界里,时间往往是一个被精心雕琢的概念。今天,让我们来探讨一下,如果从零开始做起来一家装饰画店铺,究竟需要多久的时间。   我们要明确“店铺”的定义。在这里,它不仅仅是一个物理空间,更是一个品牌理念、设计风格与市场定位的综合体。因此,准备工作涉及到市场调研、品牌策划、设计理念的形成等多个方面。这一阶

聚鼎科技:新人开一家装饰画店铺怎么快速起店

在当下这个看重审美和个性表达的时代,开设一家装饰画店铺无疑是迎合市场的明智选择。对于新人来说,快速且有效地启动一家装饰画店铺并非易事,但通过遵循一些关键步骤,可以大大缩短起步时间并提高成功率。   进行市场调研,明确目标消费群体的偏好与需求。可以通过网络调查、访问已有的艺术画廊或装饰画店,甚至是直接与潜在客户交流来收集信息。了解消费者的喜好后,选择适合的装饰画风格和价格区间,确保产品能够满足

冒雨上班,公司没了,又一家神仙公司撤出

伤心 IBM 上周五(8月23号),在 IBM 工作的同学还在正常上下班,突然发现访问权限被收回了。 后来消息被多方证实,IBM 中国的所有研发和测试岗位的权限均被收回,没有一点的征兆或信号,不少同学当时还正在加班。 涉及的员工主要集中在 IBM 中国开发中心(CDL)与 IBM 中国系统中心(CSL),这两个部门是 IBM 在中国的重要研发与测试基地。 一时间,这些研发和测试的小伙伴,都

学网页平面设计培训,选择一家好机构

学网页平面设计培训,选择一家好机构     不管是网页设计也好,还是平面设计也好,学网页平面设计培训,首先选择一家好机构。那么国内又有几家给你的网页平面设计培训机构呢?     广州传智播客网页平面设计全国最牛的培训,而且性价比是全国最高的。看一个培训机构好不好,首先得瞧它的口碑好不好,广州传智播客平面设计培训教学老师就是有实战经验和教学经验的,实行全天制教学,老

选择一家正规的应急指挥中心控制台厂家有多关键

在当今社会,随着自然灾害、突发事件及安全挑战的日益复杂多变,应急管理体系的构建显得尤为重要。而应急指挥中心作为应对各类紧急情况的神经中枢,其高效运作离不开先进、可靠的控制台设备支持。在此背景下,选择一家正规的应急指挥中心控制台厂家,成为了确保应急响应速度与效率的关键一环。   嘉德立作为一家深耕应急指挥中心控制台领域的厂家,我们致力于为客户提供定制化、智能化的解决方案。我们深知,在紧急情

一家物流装备企业终止,业绩下滑严重,恐不符合创业板新上市标准

鸿安机械终止的原因如下:首先,报告期内鸿安机械的营业收入和净利润出现下滑趋势,公司最近一年净利润恐不达标,或许不能满足创业板更新的第一套上市标准;其次,鸿安机械经营业绩对比同行业可比公司,规模及增速相对较低;同时,报告期内鸿安机械收购多家企业,但其收购价格的确定依据及公允性遭交易所质疑,并且实控人配偶刘慧欣持有鸿安自动化股权期间,未实缴出资,鸿安自动化被收购时处于亏损状态,也遭交易所问询;最后

两大物流系统集成商逆风大涨,一家归母3连涨,一家海外高利润爆单

导语 大家好,我是社长,老K。专注分享智能制造和智能仓储物流等内容。 新书《智能物流系统构成与技术实践》 在物流行业的风云变幻中,今天国际与德马科技两大领军企业近日发布的2024年中期业绩报告,犹如两股强劲的东风,吹散了市场的阴霾。 报告显示,这两家企业在复杂多变的市场环境中,不仅实现了营收与利润的双增长,更以亮眼的业绩展现了其强大的市场适应力与竞争力。 今天国际:稳健增长,三连跳 步入八月下旬