OpenGL-顶点数组 缓冲区

一个立方体有六个面,每个面是一个正方形,好,绘制六个正方形就可以了。
glBegin(GL_QUADS);
	glVertex3f(...);
	glVertex3f(...);
	glVertex3f(...);
	glVertex3f(...);
glEnd();
为了绘制六个正方形,我们为每个正方形指定四个顶点,最终我们需要指定 6*4=24 
个顶点。但是我们知道,一个立方体其实总共只有八个顶点,要指定 24 次,就意味
着每个顶点其实重复使用了三次。

如果我们定义一个数组,把八个顶点都放到数组里,然后每次指定顶点都使用指针,
而不是使用直接的数据,这样就避免了在指定顶点时考虑大量的数据,于是减少了代
码出错的可能性。
// 将立方体的八个顶点保存到一个数组里面
static const GLfloat vertex_list[][3] = {
	-0.5f, -0.5f, -0.5f,
	0.5f, -0.5f, -0.5f,
	// ...
};
// 指定顶点时,用指针,而不用直接用具体的数据
glBegin(GL_QUADS);
glVertex3fv(vertex_list[0]);
glVertex3fv(vertex_list[2]);
glVertex3fv(vertex_list[3]);
glVertex3fv(vertex_list[1]);
// ...
glEnd();

/*
	修改之后,虽然代码变长了,但是确实易读得多。很容易就看出第 0, 2, 3, 1 这四个顶点构成一个正方形。
	稍稍观察就可以发现,我们使用了大量的 glVertex3fv 函数,其实每一句	
	都只有其中的顶点序号不一样,因此我们可以再定义
	一个序号数组,把所有的序号也放进去。这样一来代码就更加简单了。
*/

// 将立方体的八个顶点保存到一个数组里面
static const GLfloat vertex_list[][3] = {
	-0.5f, -0.5f, -0.5f,
	0.5f, -0.5f, -0.5f,
	-0.5f, 0.5f, -0.5f,
	0.5f, 0.5f, -0.5f,
	-0.5f, -0.5f, 0.5f,
	0.5f, -0.5f, 0.5f,
	-0.5f, 0.5f, 0.5f,
	0.5f, 0.5f, 0.5f,
};
// 将要使用的顶点的序号保存到一个数组里面
static const GLint index_list[][4] = {
	0, 2, 3, 1,
	0, 4, 6, 2,
	0, 1, 5, 4,
	4, 5, 7, 6,
	1, 3, 7, 5,
	2, 6, 7, 3,
};
// 绘制的时候代码很简单
glBegin(GL_QUADS);
for(i=0; i<6; ++i) // 有六个面,循环六次
{
	for(j=0; j<4; ++j) // 每个面有四个顶点,循环四次
	{
		glVertex3fv(vertex_list[index_list[i][j]]);
	}
}
glEnd();
/*
	这样,我们就得到一个比较成熟的绘制立方体的版本了。它的数据和程序
	代码基本上是分开的,所有的顶点放到一个数组中,
	使用顶点的序号放到另一个数组中,而利用这两个数组来绘制立方体的代
	码则很简单
*/
/*
	在绘制之前调用如下的代码:
		glFrontFace(GL_CCW);
		glCullFace(GL_BACK);
		glEnable(GL_CULL_FACE); // 开启剔除操作效果
		glPolygonMode(GL_FRONT_AND_BACK, GL_LINE);
		则绘制出来的图形就只有正面,并且只显示边线,不进行填充。
*/

顶点数组

问题:
	前面的方法中,我们将数据和代码分离开,看起来只要八个顶点就可以绘制一个立
	方体了。但是实际上,循环还是执行了 6*4=24 次,也就是说虽然代码的结构清晰
	了不少,但是程序运行的效率,还是和最原始的那个方法一样。减少函数的调用次
	数,是提高运行效率的方法之一。

	于是我们想到了显示列表。把绘制立方体的代码装到一个显示列表中,以后只要调
	用这个显示列表即可。,但是显示列表有一个缺点,那就是一旦建立后不可再改。
	如果我们要绘制的不是立方体,而是一个能够走动的人物,因为人物走动时,四肢
	的位置不断变化,几乎没有办法把所有的内容装到一个显示列表中。必须每种动作
	都使用单独的显示列表,这样会导致大量的显示列表管理困难。

	顶点数组是解决这个问题的一个方法。使用顶点数组的时候,也是像前面的方法一
	样,用一个数组保存所有的顶点,用一个数组保存顶点的序号。但最后绘制的时
	候,不是编写循环语句逐个的指定顶点了,而是通知 OpenGL,“保存顶点的数组”
	和“保存顶点序号的数组”所在的位置,由 OpenGL 自动的找到顶点,并进行绘制。
// 下面的代码说明了顶点数组是如何使用的:
glEnableClientState(GL_VERTEX_ARRAY);
glVertexPointer(3, GL_FLOAT, 0, vertex_list);
glDrawElements(GL_QUADS, 24, GL_UNSIGNED_INT, index_list);
其中:
glEnableClientState(GL_VERTEX_ARRAY); 表示启用顶点数组。

glVertexPointer(3, GL_FLOAT, 0, vertex_list); 指定顶点数组的位置
3 表示每个顶点由三个量构成(x, y, z),
GL_FLOAT 表示每个量都是一个 GLfloat 类型的值,
第三个参数 0,
最后的 vertex_list 指明了数组实际的位置。

glDrawElements(GL_QUADS, 24, GL_UNSIGNED_INT, index_list); 根据序号数组
中的序号,查找到相应的顶点,并完成绘制。
GL_QUADS 表示绘制的是四边形,24 表示总共有 24 个顶点,GL_UNSIGNED_INT 
表示序号数组内每个序号都是一个 GLuint类型的值,index_list 指明了序号数组实际	
的位置。

上面三行代码代替了原来的循环。可以看到,原来的 glBegin/glEnd 不再需要了,也
不需要调用 glVertex*系列函数来指定顶点,因此可以明显的减少函数调用次数。另
外,数组中的内容可以随时修改,比显示列表更加灵活。

说明:
顶点数组实际上是多个数组,顶点坐标、纹理坐标、法线向量、顶点颜色等等,顶点
的每一个属性都可以指定一个数组,然后用统一的序号来进行访问。比如序号 3,就
表示取得颜色数组的第 3 个元素作为颜色、取得纹理坐标数组的第 3 个元素作
为纹理坐标、取得法线向量数组的第 3 个元素作为法线向量、取得顶点坐标数组的第 	
3 个元素作为顶点坐标。把所有的数据综合起来,最终得到一个顶点。

可以用 glEnableClientState/glDisableClientState 单独的开启和关闭每一种数组。
glEnableClientState(GL_VERTEX_ARRAY);
glEnableClientState(GL_COLOR_ARRAY);
glEnableClientState(GL_NORMAL_ARRAY);
glEnableClientState(GL_TEXTURE_COORD_ARRAY);

用以下的函数来指定数组的位置:
glVertexPointer
glColorPointer
glNormalPointer
glTexCoordPointer

为什么不使用原来的 glEnable/glDisable 函数,而要专门的规定一个 
glEnableClientState/glDisableClientState 函数呢?
这跟OpenGL 的工作机制有关。
OpenGL 在设计时,认为可以将整个 OpenGL 系统分为两部分,
一部分是客户端,它负责发送 OpenGL命令。
一部分是服务端,它负责接收 OpenGL 命令并执行相应的操作。

对于个人计算机来说,可以将 CPU、内存等硬件,以及用户编写的 OpenGL 程序看
做客户端,而将 OpenGL 驱动程序、显示设备等看做服务端。

通常,所有的状态都是保存在服务端的,便于 OpenGL 使用。

例如,是否启用了纹理,服务端在绘制时经常需要知道这个状态,而我们编写的客户
端 OpenGL 程序只在很少的时候需要知道这个状态。所以将这个状态放在服务端是比
较有利的。

但顶点数组的状态则不同。我们指定顶点,实际上就是把顶点数据从客户端发送到服
务端。
是否启用顶点数组,只是控制发送顶点数据的方式而已。
服务端只管接收顶点数据,而不必管顶点数据到底是用哪种方式指定的(可以直接使
用glBegin/glEnd/glVertex*,也可以使用顶点数组)。
所以,服务端不需要知道顶点数组是否开启。因此,顶点数组的状态放在
客户端是比较合理的。

为了表示服务端状态和客户端状态的区别,服务端的状态用 glEnable/glDisable,
客户端的状态则用glEnableClientState/glDisableClientState	


stride 参数
顶点数组并不要求所有的数据都连续存放。
如果数据没有连续存放,则指定数据之间的间隔即可。

例如:我们使用一个 struct 来存放顶点中的数据。注意每个顶点除了坐标外,还有额外的数据(这里是一个 int 类型的值)。
typedef struct __point__ {
	GLfloat position[3];
	int id;
} Point;

Point vertex_list[] = {
	-0.5f, -0.5f, -0.5f, 1,
	0.5f, -0.5f, -0.5f, 2,
	-0.5f, 0.5f, -0.5f, 3,
	0.5f, 0.5f, -0.5f, 4,
	-0.5f, -0.5f, 0.5f, 5,
	0.5f, -0.5f, 0.5f, 6,
	-0.5f, 0.5f, 0.5f, 7,
	0.5f, 0.5f, 0.5f, 8,
};

static GLint index_list[][4] = {
	0, 2, 3, 1,
	0, 4, 6, 2,
	0, 1, 5, 4,
	4, 5, 7, 6,
	1, 3, 7, 5,
	2, 6, 7, 3
};

glEnableClientState(GL_VERTEX_ARRAY);
glVertexPointer(3, GL_FLOAT, sizeof(Point), vertex_list);
glDrawElements(GL_QUADS, 24, GL_UNSIGNED_INT, index_list);

/*
	注意最后三行代码,可以看到,几乎所有的地方都和原来一样,只在 
	glVertexPointer 函数的第三个参数有所不同。

	这个参数就是 stride,它表示“从一个数据的开始到下一个数据的开始,
	所相隔的字节数”。
	这里设置为 sizeof(Point)就刚刚好。
	如果设置为 0,则表示数据是紧密排列的,对于 3 个 GLfloat 的情况,
	数据紧密排列时 stride 实际上为 3*4=12。
*/
混合数组。如果需要同时使用颜色数组、顶点坐标数组、纹理坐标数组、等等,有一
种方式是把所有的数据都混合起来,指定到同一个数组中。这就是混合数组。
GLfloat arr_c3f_v3f[] = {
	1, 0, 0, 0, 1, 0,
	0, 1, 0, 1, 0, 0,
	0, 0, 1, -1, 0, 0,
};
GLuint index_list[] = {0, 1, 2};
glInterleavedArrays(GL_C3F_V3F, 0, arr_c3f_v3f);
glDrawElements(GL_TRIANGLES, 3, GL_UNSIGNED_INT, index_list);
/*
	glInterleavedArrays,可以设置混合数组。

	这个函数会自动调用 glVertexPointer, glColorPointer 等函数,并且自
	动的开启或禁用相关的数组。

	函数的第一个参数表示了混合数组的类型。例如 GL_C3F_V3F 表示:三个
	浮点数作为颜色、三个浮点数作为顶点坐标。

	也可以有其它的格式,比如
	GL_V2F, GL_V3F,GL_C4UB_V2F, GL_C4UB_V3F, GL_C3F_V3F, GL_N3F_V3F, 
	GL_C4F_N3F_V3F, GL_T2F_V3F,GL_T4F_V4F, GL_T2F_C4UB_V3F, GL_T2F_C3F_V3F, GL_T2F_N3F_V3F, GL_T2F_C4F_N3F_V3F,GL_T4F_C4F_N3F_V4F

	其中 T 表示纹理坐标,C 表示颜色,N 表示法线向量,V 表示顶点坐标。
*/
顶点数组与显示列表的区别:
两者都可以明显的减少函数的调用次数。
对于顶点数组,顶点数据是存放在内存中的,也就是存放在客户端。每次绘制的时
候,需要把所有的顶点数据从客户端(内存)发送到服务端(显示设备),然后进行
处理。
对于显示列表,顶点数据是放在显示列表中的,显示列表本身又是存放在服务器端
的,所以不会重复的发送数据。
对于顶点数组,因为顶点数据放在内存中,所以可以随时修改,每次绘制的时候都会
把当前数组中的内容作为顶点数据发送并进行绘制。
对于显示列表,数据已经存放到服务器段,并且无法取出,所以无法修改。

也就是说,显示列表可以避免数据的重复发送,效率会较高;顶点数组虽然会重复的
发送数据,但由于数据可以随时修改,灵活性较好。

顶点缓冲区对象

	顶点缓冲区对象是OpenGL1.5 所提供的功能,成为标准前是一个ARB 扩展。
	可以通过 GL_ARB_vertex_buffer_object 扩展来使用这项功能。
	ARB 扩展的函数名称以ARB 结尾,常量名称以字母_ARB 结尾,标准函数、常量
	去掉了ARB字样。
	OpenGL实现同时支持 vertex buffer object 的标准版 和ARB 扩展板。
	
	
	顶点缓冲区对象,它数据存放在服务端,同时也允许客户端灵活的修改,兼顾运行
	效率和灵活性

顶点缓冲区对象跟纹理有很多相似之处。首先,分配一个缓冲区对象编号,然后,
为对象编号的缓冲对象指定数据,以后可以随时修改其中的数据。
纹理对象顶点缓冲区对象
分配编号glGenTexturesglGenBuffersARB
绑定(指定为当前所使用的对象)glBindTextureglBindBufferARB
指定数据glTexImage*glBufferDataARB
修改数据glTexSubImage*glBufferSubDataARB
顶点数据和序号各自使用不同的缓冲区,就是顶点数据存放在 GL_ARRAY_BUFFER_ARB 类型的缓冲区中,序号数据放在 GL_ELEMENT_ARRAY_BUFFER_ARB 类型的缓冲区中。
static GLuint vertex_buffer;
static GLuint index_buffer;

// 分配一个缓冲区,并将顶点数据指定到其中
glGenBuffersARB(1,&vertex_buffer);
glBindBufferARB(GL_ARRAY_BUFFER_ARB,vertex_buffer);
glBufferDataARB(GL_ARRAY_BUFFER_ARB,sizeof(vertex_list),vertex_list,GL_STATIC_DRAW_ARB);

// 分配一个缓冲区,并将序号数据指定到其中
glGenbuffersARB(1,&index_buffer);
glBindBufferARB(GL_ELEMENT_ARRAY_BUFFER_ARB,index_buffer);
glBufferDataARB(GL_ELEMENT_ARRAY_BUFFER_ARB,sizeof(index_list),index_list,GL_STATIC_DRAW_ARB);
指定缓冲区数据时,最后一个参数是关于性能的提示。
共有  STREAM_DRAW, STREAM_READ, STREAM_COPY, 
STATIC_DRAW,STATIC_READ, STATIC_COPY, DYNAMIC_DRAW, 
DYNAMIC_READ, DYNAMIC_COPY
每一种都表示了使用频率和用途。

使用glBindBufferARB 后,各种使用指针为参数的 OpenGL 函数,行为会发生变化。
以 glColor3f 为例,通常,这个函数接受一个指针作为参数,从指针所指的位置取出
连续的三个浮点数,作为当前的颜色。

当使用 glBindBufferARB后,这个函数不再从指针所指的位置取数据。

函数会先把指针转化为整数,假设转化后结果为k,则会从当前缓冲区的第K个字节开
始取数据。 特别一点,如果我们写 glColor3fv(NULL);因为NULL转化为整数后通常
是零,所以从缓冲区的第0个字节开始取数据,也就是从缓冲区最开始的位置取数
据。
glVertexPointer(3,GL_FLOAT,0,vertex_list);
glDrawElements(GL_QUADS,24,GL_UNSIGNED_INT,index_list);
// 使用了缓冲区对象后,就变成
glVertexPointer(3,GL_FLOAT,0,NULL);
glDrawElements(GL_QUADS,24,GL_UNSIGNED_INT,NULL);

// 以下是完整 的使用了 顶点缓冲区对象的代码
static GLfloat vertex_list[][3] = {
	-0.5f, -0.5f, -0.5f,
	0.5f, -0.5f, -0.5f,
	-0.5f, 0.5f, -0.5f,
	0.5f, 0.5f, -0.5f,
	-0.5f, -0.5f, 0.5f,
	0.5f, -0.5f, 0.5f,
	-0.5f, 0.5f, 0.5f,
	0.5f, 0.5f, 0.5f,
}
static GLint index_list[][4] = {
	0, 2, 3, 1,
	0, 4, 6, 2,
	0, 1, 5, 4,
	4, 5, 7, 6,
	1, 3, 7, 5,
	2, 6, 7, 3,
};
if(GLEE_ARB_vertex_buffer_object)
{ 
	// 如果支持顶点缓冲区对象
	static int isFirstCall = 1;
	static GLuint vertex_buffer;
	static GLuint index_buffer;
	if(isFirstCall)
	{
		// 第一次调用时,初始化缓冲区
		isFirstCall = 0;
		// 分配一个缓冲区,并将顶点数据指定到其中
		glGenBuffersARB(1,&vertex_buffer);
		glBindBufferARB(GL_ARRAY_BUFFER_ARB,vertex_buffer);
		glBufferDataARB(GL_ARRAY_BUFFER_ARB,sizeof(vertex_list),vertex_list,GL_STATIC_DRAW_ARB);

		// 分配一个缓冲区,并将序号数据指定到其中
		glGenBuffersARB(1,&index_buffer);
		glBindBufferARB(GL_ELEMENT_ARRAY_BUFFER_ARB,index_buffer);
		glBufferDataARB(GL_ELEMENT_ARRAY_BUFFER_ARB,sizeof(index_list),index_list,GL_STATIC_DRAW_ARB);
	}
	glBindBufferARB(GL_ARRAY_BUFFER_ARB,vertex_buffer);
	glBindBUfferARB(GL_ELEMENT_ARRAY_BUFFER_ARB,index_buffer);

	// 实际使用时与顶点数组非常相似,只是在指定数组时不再指定实际的数组,改为指定NULL 
	glEnableClientState(GL_VERTEX_ARRAY);
	glVertexPointer(3,GL_FLOAT,0,NULL);
	glDrawElements(GL_QUADS,24,GL_UNSIGNED_INT,NULL);
}
else
{
	// 不支持顶点缓冲区对象
	// 使用顶点数组
	glEnableClientState(GL_VERTEX_ARRAY);
	glVertexPointer(3,GL_FLOAT,0,vertex_list);
	glDrawElements(GL_QUADS,24,GL_UNSIGNED_INT,index_list);
}
/*
	可以分配多个缓冲区对象,顶点坐标,颜色、纹理坐标等数据,可以各自
	单独使用一个缓冲区。
	每个缓冲区可以有不同的性能提示,比如在绘制一个运动的人物时,顶点
	坐标数据经常变化,但法线向量、纹理坐标等则不
    会变化,可以给予不同的性能提示,以提高性能。
*/
绘制物体的时候,应该将数据单独存放,尽量不要到处写类似 glVertex3f(1.0f, 0.0f, 
1.0f)这样的代码。

将顶点坐标、顶点序号都存放到单独的数组中,可以让绘制的代码变得简单。可以把
绘制物体的所有命令装到一个显示列表中,这样可以避免重复的数据传送。但是因为
显示列表一旦建立,就无法修改,所以灵活性很差。
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值