使用了顶点缓冲技术后,绘制效率有了较大的提升。但是还有一点不尽如人意,那就是顶点的位置坐标、法向量、纹理坐标等不同方面的数据每次使用时需要单独指定,重复了一些不必要的工作。WebGL2提供了一种专门用于解决此问题的对象——顶点数组对象(VAO)。本节将介绍顶点数组对象。

顶点数组对象,在WebGL1中,是一个扩展对象,该扩展对象的名称是OES_vertex_array_object;而在WebGL2中可以直接使用;如果你在WebGL1中已经使用过OES_vertex_array_object,那么你只需要了解在WebGL2和WebGL1的调用方式的差异即可

下面会对顶点数组对象做详细的介绍。

顶点数组对象

顶点数组对象( VAO )是这样一种对象: 它封装了与顶点处理器有关的所有数据,它记录了顶点缓存区和索引缓冲区的引用,以及顶点的各种属性的布局而不是实际的数据。

顶点数组对象的优点

这样做的优点是: 一旦为一个 对象指定了一个VAO之后,可以ton通过对该VAO对象进行简单的绑定操作来导入对象的所有引用和状态。在之后绘制对象时候,不需要在手动来导入对象的引用和状态,VAO替你记住了它。
通过VAO可以简化缓冲区的绑定过程,即可以减少代码的调用次数,也提升了WebGL状态切换的效率。

案例:用顶点数组对象绘制两个三角形

下面通过代码来说明顶点数组对象的使用,本案例代码绘制两个顶点色的三角形,最终显示的效果如下:

var triangleArray = gl.createVertexArray();
gl.bindVertexArray(triangleArray); var positions = new Float32Array([
-0.5, -0.5, 0.0,
0.0, -0.5, 0.0,
0.0, 0.0, 0.0
]);
var positionBuffer = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, positionBuffer);
gl.bufferData(gl.ARRAY_BUFFER, positions, gl.STATIC_DRAW);
gl.vertexAttribPointer(0, 3, gl.FLOAT, false, 0, 0);
gl.enableVertexAttribArray(0);
var colors = new Float32Array([
1.0, 0.0, 0.0,
0.0, 1.0, 0.0,
0.0, 0.0, 1.0
]);
var colorBuffer = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, colorBuffer);
gl.bufferData(gl.ARRAY_BUFFER, colors, gl.STATIC_DRAW);
gl.vertexAttribPointer(1, 3, gl.FLOAT, false, 0, 0);
gl.enableVertexAttribArray(1); var triangleArray2 = gl.createVertexArray();
gl.bindVertexArray(triangleArray2); var positions2 = new Float32Array([
0.0, -0.0, 0.0,
0.5, 0.0, 0.0,
0.0, 0.5, 0.0
]);
var positionBuffer2 = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, positionBuffer2);
gl.bufferData(gl.ARRAY_BUFFER, positions2, gl.STATIC_DRAW);
gl.vertexAttribPointer(0, 3, gl.FLOAT, false, 0, 0);
gl.enableVertexAttribArray(0);
var colors2 = new Float32Array([
1.0, 1.0, 0.0,
0.0, 1.0, 1.0,
0.0, 1.0, 1.0
]);
var colorBuffer2 = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, colorBuffer2);
gl.bufferData(gl.ARRAY_BUFFER, colors2, gl.STATIC_DRAW);
gl.vertexAttribPointer(1, 3, gl.FLOAT, false, 0, 0);
gl.enableVertexAttribArray(1);
////////////////
// DRAW
////////////////
gl.clear(gl.COLOR_BUFFER_BIT); gl.bindVertexArray(triangleArray);
gl.drawArrays(gl.TRIANGLES, 0, 3); gl.bindVertexArray(triangleArray2);
gl.drawArrays(gl.TRIANGLES, 0, 3);

定义三角形相关数据和缓冲区

首先,定义了三角形的顶点数据和缓冲区和WebGL1的代码很类似,下面是一个三角相关数据定义的代码

var triangleArray = gl.createVertexArray();
gl.bindVertexArray(triangleArray); var positions = new Float32Array([
-0.5, -0.5, 0.0,
0.0, -0.5, 0.0,
0.0, 0.0, 0.0
]);
var positionBuffer = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, positionBuffer);
gl.bufferData(gl.ARRAY_BUFFER, positions, gl.STATIC_DRAW);
gl.vertexAttribPointer(0, 3, gl.FLOAT, false, 0, 0);
gl.enableVertexAttribArray(0);
var colors = new Float32Array([
1.0, 0.0, 0.0,
0.0, 1.0, 0.0,
0.0, 0.0, 1.0
]);
var colorBuffer = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, colorBuffer);
gl.bufferData(gl.ARRAY_BUFFER, colors, gl.STATIC_DRAW);
gl.vertexAttribPointer(1, 3, gl.FLOAT, false, 0, 0);
gl.enableVertexAttribArray(1);

可以看出除了前两行 创建VAO和绑定VAO代码外,其他的代码都是WebGL1一样的代码:

    • 定义坐标数组
    • 创建顶点坐标缓冲区
    • 绑定缓冲区并填充缓冲区数据
    • 把缓冲区分配给attribute变量
    • 启用attribute变量

    代码中定义了两种顶点信息:顶点坐标和顶点颜色

    创建另外一个三角形的相关数据的代码和第一个类似,不重复说明。

    VAO对象在创建顶点数据的作用

    现在回到前面两行代码:

    var triangleArray = gl.createVertexArray();
    gl.bindVertexArray(triangleArray);

    第一行代码创建了一个VAO对象,第二行代码绑定该VAO对象,这两行代码的作用是: 后面关于顶点缓冲对象的操作和状态,都会被记录到这个VAO对象中,以后绘制的时候,只需要调用gl.bindVertexArray方法,绑定该对象,就会自动使用相关的状态。

    VAO 对象在绘制的时候的作用

    下面在看绘制的代码

           gl.clear(gl.COLOR_BUFFER_BIT);// 清空颜色缓冲区
    // 绘制第一个三角形
    gl.bindVertexArray(triangleArray);
    gl.drawArrays(gl.TRIANGLES, 0, 3);
    // 绘制第二个三角形
    gl.bindVertexArray(triangleArray2);
    gl.drawArrays(gl.TRIANGLES, 0, 3);

    第一行代码,清空颜色缓冲区,和WebGL1一样;然后绘制第一个三角形,绘制时候,

    • 先调用 gl.bindVertexArray(triangleArray)把第一个三角形相关的缓冲区状态恢复,
    • 然后调用gl.drawArrays(gl.TRIANGLES, 0, 3)绘制

    绘制第二个三角形和第一个三角形类似;
    回顾下,如果不使用顶点数组对象,绘制第一个三角形的代码便是这样:

            gl.bindBuffer(gl.ARRAY_BUFFER, positionBuffer);
    gl.vertexAttribPointer(0, 3, gl.FLOAT, false, 0, 0);
    gl.enableVertexAttribArray(0);
    gl.bindBuffer(gl.ARRAY_BUFFER, colorBuffer);
    gl.vertexAttribPointer(1, 3, gl.FLOAT, false, 0, 0);
    gl.enableVertexAttribArray(1); gl.drawArrays(gl.TRIANGLES, 0, 3);

    绘制第二个三角形类似,由此可以看出,使用VAO对象之后,gl.bindVertexArray这一行代码相当于以下代码:

            gl.bindBuffer(gl.ARRAY_BUFFER, positionBuffer);
    gl.vertexAttribPointer(0, 3, gl.FLOAT, false, 0, 0);
    gl.enableVertexAttribArray(0);
    gl.bindBuffer(gl.ARRAY_BUFFER, colorBuffer);
    gl.vertexAttribPointer(1, 3, gl.FLOAT, false, 0, 0);
    gl.enableVertexAttribArray(1);

    也就是绑定缓冲区对象,分配attribute和启用attribute变量等等操作都可以在绘制的时候可以不再调用了。 有次可以看出,使用VAO对象的好处:

    • 减少代码量,提高开发效率
    • 提高绘制效率(因为减少了WebGL相关函数的调用,并且WebGL内部对VAO进行了优化)

    WebGL1中如何使用VAO

    在WebGL1.0中VAO是通过扩展方式提供的,首先需要获取对应的扩展对象:

    var ext = gl.getExtension("OES_vertex_array_object");

    如果返回的ext位null说明浏览器不支持该扩展。
    有了上面的ext对象,便可以创建VAO了:

    var vao = ext.createVertexArrayOES();

    有了VAO对象之后,就可以进行绑定操作:

    // bind
    ext.bindVertexArrayOES(vao);
    // unbind
    ext.bindVertexArrayOES(null);

    WebGL2系列之顶点数组对象的更多相关文章

    1. Opengl_入门学习分享和记录_03_渲染管线(三)借助顶点数组对象VAO提高绑定属性效率

      目前我们已经知道了,如果想要顶点着色器解释理解我们的输入数据,就必须要按照以下繁琐的步骤:第一步:将输入的数据复制一份到缓冲区,供OpenGL使用.而这块新出现的区域由VBO管理和表示.(若有多个输入 ...

    2. 3D Computer Grapihcs Using OpenGL - 19 Vertex Array Object(顶点数组对象)

      大部分OpenGL教程都会在一开始就讲解VAO,但是该教程的作者认为这是很不合理的,因为要理解它的作用需要建立在我们此前学过的知识基础上.因此直到教程已经进行了一大半,作者才引入VAO这个概念.在我看 ...

    3. WebGL2系列之实例数组(Instanced Arrays)

      实例化数组 实例化是一种只调用一次渲染函数却能绘制出很多物体的技术,它节省渲染一个物体时从CPU到GPU的通信时间.实例数组是这样的一个对象,使用它,可以把原来的的uniform变量转换成attrib ...

    4. OpenGL中glVertex、显示列表(glCallList)、顶点数组(Vertex array)、VBO及VAO区别

      OpenGL中glVertex.显示列表(glCallList).顶点数组(Vertex array).VBO及VAO区别 1.glVertex 最原始的设置顶点方法,在glBegin和glEnd之间 ...

    5. MVVM架构~knockoutjs系列之为Ajax传递Ko数组对象

      返回目录 一些要说的 这是一个很有意思的题目,在KO里,有对象和数组对象两种,但这两种对象对外表现都是一个function,如果希望得到他的值,需要进行函数式调用,如ko_a(),它的结果为一个具体值 ...

    6. 顶点缓存对象(VBO)

      创建VBO 绘制VBO 更新VBO 实例 GL_ARB_vertex_buffer_object扩展致力于提供顶点数组与显示列表的优势来提升OpenGL效率,同时避免它们实现上的不足.顶点缓存对象(V ...

    7. OpenGL顶点数组

      概述 作为在立即模式(glBegin()与glEnd()之间)下指定单个顶点数据的替代,你可以保存顶点数据在一组列表中,包括顶点位置.法线.纹理坐标与颜色信息.并且你可以通过索引数组解引用数组元素绘制 ...

    8. OpenGL顶点缓冲区对象(VBO)

      转载 http://blog.csdn.net/dreamcs/article/details/7702701 创建VBO        GL_ARB_vertex_buffer_object 扩展可 ...

    9. openGL 提升渲染性能 之 顶点数组 VBO IBO VAO

      使用openGL图形库绘制,都需要通过openGL接口向图像显卡提交顶点数据,显卡根据提交的数据绘制出相应的图形. openGL绘制方式有:直接模式,显示列表,顶点数组,顶点索引. 直接模式:最简单, ...

    随机推荐

    1. Lesson A puma at large

      新概念三 Lesson 1 A puma at large 词汇: 1. spot 易混淆: recognize v. [认出], identify v. [识别sb/sth的身份] v. 看出,发现 ...

    2. linux中mariadb用navicat远程连接

      在Linux中创建数据库并且远程图形化工具连接 安装数据库 [root@node1 ~]# yum install mariadb-server -y #这里我使用的mariadb 其他数据库也可以 ...

    3. layui 表单的使用

      <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...

    4. gitee 的使用

      Git安装教程(windows)   Git是当今最流行的版本控制软件,它包含了许多高级工具,这里小编就讲一下Git的安装. 下载地址:https://git-scm.com/downloads 首先 ...

    5. 利用mysqldump 与 nginx定时器 定时备份mysql库

      1.安装mysqldump(如果备份远程mysql库,本地不用安装mysql 也可以单独使用) yum -y install holland-mysqldump.noarch 2.编写备份脚本 首先这 ...

    6. tensorflow源码解析之framework-op

      目录 什么是op op_def定义 op注册 op构建与注册辅助结构 op重写 关系图 涉及的文件 迭代记录 1. 什么是op op和kernel是TF框架中最重要的两个概念,如果一定要做一个类比的话 ...

    7. java基础复习记录

      java基础复习记录(数组.对象.异常) 数组 数组的定义 数组是相同类型数据的有序集合.如:篮球队就是一个数组,队员球服上有号码,相当于索引.通过某一个的某一个号码来确认是某一个队员.数组中索引从0 ...

    8. SQL语句的使用,SELECT - 从数据库表中获取数据 UPDATE - 更新数据库表中的数据 DELETE - 从数据库表中删除数据 INSERT INTO - 向数据库表中插入数据

      SQL DML 和 DDL 可以把 SQL 分为两个部分:数据操作语言 (DML) 和 数据定义语言 (DDL). SQL (结构化查询语言)是用于执行查询的语法. 但是 SQL 语言也包含用于更新. ...

    9. [差分数组] LeetCode789 得分最高的最小轮调

      LeetCode 得分最高的最小轮调 今天当然CV了因为今天比较忙,所以直接走算法,因为什么都不做的话并不符合社会主义核心价值观,今天小学一手查分数组. 题目:并不存在CV了还写什么题解 算法背景: ...

    10. python 装饰器的使用

      装饰器本质上是一个Python函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值也是一个函数对象.它经常用于有切面需求的场景,比如:插入日志.性能测试.事务处理.缓存.权 ...