`

【转】OpenGL入门学习——第七课

 
阅读更多

今天介绍关于OpenGL显示列表的知识。本课内容并不多,但需要一些理解能力。在学习时,可以将显示列表与C语言的“函数”进行类比,加深体会。

 

我 们已经知道,使用OpenGL其实只要调用一系列的OpenGL函数就可以了。然而,这种方式在一些时候可能导致问题。比如某个画面中,使用了数千个多边 形来表现一个比较真实的人物,OpenGL为了产生这数千个多边形,就需要不停的调用glVertex*函数,每一个多边形将至少调用三次(因为多边形至 少有三个顶点),于是绘制一个比较真实的人物就需要调用上万次的glVertex*函数。更糟糕的是,如果我们需要每秒钟绘制60幅画面,则每秒调用的 glVertex*函数次数就会超过数十万次,乃至接近百万次。这样的情况是我们所不愿意看到的。
同时,考虑这样一段代码:

 

const int segments = 100;
const GLfloat pi = 3.14f;
int i;
glLineWidth(10.0);
glBegin(GL_LINE_LOOP);
for(i=0; i<segments; ++i)
{
    GLfloat tmp = 2 * pi * i / segments;
    glVertex2f(cos(tmp), sin(tmp));
}
glEnd();

 

 

 


这段代码将绘制一个圆环。如果我们在每次绘制图象时调用这段代码,则虽然可以达到绘制圆环的目的,但是cos、sin等开销较大的函数被多次调用,浪费了 CPU资源。如果每一个顶点不是通过cos、sin等函数得到,而是使用更复杂的运算方式来得到,则浪费的现象就更加明显。

经过分析,我们可以发现上述两个问题的共同点:程序多次执行了重复的工作,导致CPU资源浪费和运行速度的下降。使用显示列表可以较好的解决上述两个问题。
在编写程序时,遇到重复的工作,我们往往是将重复的工作编写为函数,在需要的地方调用它。类似的,在编写OpenGL程序时,遇到重复的工作,可以创建一个显示列表,把重复的工作装入其中,并在需要的地方调用这个显示列表。

 

 

使用显示列表一般有四个步骤:分配显示列表编号、创建显示列表、调用显示列表、销毁显示列表。

一、分配显示列表编号
OpenGL允许多个显示列表同时存在,就好象C语言允许程序中有多个函数同时存在。C语言中,不同的函数用不同的名字来区分,而在OpenGL中,不同的显示列表用不同的正整数来区分。
你可以自己指定一些各不相同的正整数来表示不同的显示列表。但是如果你不够小心,可能出现一个显示列表将另一个显示列表覆盖的情况。为了避免这一问题,使用glGenLists函数来自动分配一个没有使用的显示列表编号。
glGenLists函数有一个参数i,表示要分配i个连续的未使用的显示列表编号。返回的是分配的若干连续编号中最小的一个。例如,glGenLists(3);如果返回20,则表示分配了20、21、22这三个连续的编号。如果函数返回零,表示分配失败。
可以使用glIsList函数判断一个编号是否已经被用作显示列表。

二、创建显示列表
创建显示列表实际上就是把各种OpenGL函数的调用装入到显示列表中。使用glNewList开始装入,使用glEndList结束装入。 glNewList有两个参数,第一个参数是一个正整数表示装入到哪个显示列表。第二个参数有两种取值,如果为GL_COMPILE,则表示以下的内容只 是装入到显示列表,但现在不执行它们;如果为GL_COMPILE_AND_EXECUTE,表示在装入的同时,把装入的内容执行一遍。
例如,需要把“设置颜色为红色,并且指定一个坐标为(0, 0)的顶点”这两条命令装入到编号为list的显示列表中,并且在装入的时候不执行,则可以用下面的代码:

 

glNewList(list, GL_COMPILE);
glColor3f(1.0f, 0.0f, 0.0f);
glVertex2f(0.0f, 0.0f);
glEnd();

 
注意:显示列表只能装入OpenGL函数,而不能装入其它内容。例如:

 

int i = 3;
glNewList(list, GL_COMPILE);
if( i > 20 )
    glColor3f(1.0f, 0.0f, 0.0f);
glVertex2f(0.0f, 0.0f);
glEnd();

 
其中if这个判断就没有被装入到显示列表。以后即使修改i的值,使i>20的条件成立,则glColor3f这个函数也不会被执行。因为它根本就不存在于显示列表中。

另外,并非所有的OpenGL函数都可以装入到显示列表中。例如,各种用于查询的函数,它们无法被装入到显示列表,因为它们都具有返回值,而 glCallList和glCallLists函数都不知道如何处理这些返回值。在网络方式下,设置客户端状态的函数也无法被装入到显示列表,这是因为显 示列表被保存到服务器端,各种设置客户端状态的函数在发送到服务器端以前就被执行了,而服务器端无法执行这些函数。分配、创建、删除显示列表的动作也无法 被装入到另一个显示列表,但调用显示列表的动作则可以被装入到另一个显示列表。

三、调用显示列表
使用glCallList函数可以调用一个显示列表。该函数有一个参数,表示要调用的显示列表的编号。例如,要调用编号为10的显示列表,直接使用glCallList(10);就可以了。
使用glCallLists函数可以调用一系列的显示列表。该函数有三个参数,第一个参数表示了要调用多少个显示列表。第二个参数表示了这些显示列表的编 号的储存格式,可以是GL_BYTE(每个编号用一个GLbyte表示),GL_UNSIGNED_BYTE(每个编号用一个GLubyte表 示),GL_SHORT,GL_UNSIGNED_SHORT,GL_INT,GL_UNSIGNED_INT,GL_FLOAT。第三个参数表示了这些 显示列表的编号所在的位置。在使用该函数前,需要用glListBase函数来设置一个偏移量。假设偏移量为k,且glCallLists中要求调用的显 示列表编号依次为l1, l2, l3, ...,则实际调用的显示列表为l1+k, l2+k, l3+k, ...。
例如:
GLuint lists[] = {1, 3, 4, 8};
glListBase(10);
glCallLists(4, GL_UNSIGNED_INT, lists);
则实际上调用的是编号为11, 13, 14, 18的四个显示列表。
注:“调用显示列表”这个动作本身也可以被装在另一个显示列表中。

四、销毁显示列表
销毁显示列表可以回收资源。使用glDeleteLists来销毁一串编号连续的显示列表。
例如,使用glDeleteLists(20, 4);将销毁20,21,22,23这四个显示列表。

 

使用显示列表将会带来一些开销,例如,把各种动作保存到显示列表中会占用一定数量的内存资源。但如果使用得当,显示列表可以提升程序的性能。这主要表现在以下方面:
1、明显的减少OpenGL函数的调用次数。如果函数调用是通过网络进行的(Linux等操作系统支持这样的方式,即由应用程序在客户端发出OpenGL请求,由网络上的另一台服务器进行实际的绘图操作),将显示列表保存在服务器端,可以大大减少网络负担。
2、保存中间结果,避免一些不必要的计算。例如前面的样例程序中,cos、sin函数的计算结果被直接保存到显示列表中,以后使用时就不必重复计算。
3、便于优化。我们已经知道,使用glTranslate*、glRotate*、glScale*等函数时,实际上是执行矩阵乘法操作,由于这些函数经 常被组合在一起使用,通常会出现矩阵的连乘。这时,如果把这些操作保存到显示列表中,则一些复杂的OpenGL版本会尝试先计算出连乘的一部分结果,从而 提高程序的运行速度。在其它方面也可能存在类似的例子。
同时,显示列表也为程序的设计带来方便。我们在设置一些属性时,经常把一些相关的函数放在一起调用,(比如,把设置光源的各种属性的函数放到一起)这时,如果把这些设置属性的操作装入到显示列表中,则可以实现属性的成组的切换。
当然了,即使使用显示列表在某些情况下可以提高性能,但这种提高很可能并不明显。毕竟,在硬件配置和大致的软件算法都不变的前提下,性能可提升的空间并不大。

 


 

显示列表的内容就是这么多了,下面我们看一个例子。
假设我们需要绘制一个旋转的彩色正四面体,则可以这样考虑:设置一个全局变量angle,然后让它的值不断的增加(到达360后又恢复为0,周而复始)。 每次需要绘制图形时,根据angle的值进行旋转,然后绘制正四面体。这里正四面体采用显示列表来实现,即把绘制正四面体的若干OpenGL函数装到一个 显示列表中,然后每次需要绘制时,调用这个显示列表即可。
将正四面体的四个顶点颜色分别设置为红、黄、绿、蓝,通过数学计算,将坐标设置为:
(-0.5, -5*sqrt(5)/48,  sqrt(3)/6),
( 0.5, -5*sqrt(5)/48,  sqrt(3)/6),
(   0, -5*sqrt(5)/48, -sqrt(3)/3),
(   0, 11*sqrt(6)/48,          0)
2007年4月24日修正:以上结果有误,通过计算AB, AC, AD, BC, BD, CD的长度,发现AD, BD, CD的长度与1.0有较大偏差。正确的坐标应该是:
   A点:(  0.5,   -sqrt(6)/12, -sqrt(3)/6)
   B点:( -0.5,   -sqrt(6)/12, -sqrt(3)/6)
   C点:(    0,   -sqrt(6)/12,  sqrt(3)/3)
   D点:(    0,    sqrt(6)/4,           0)
   程序代码中也做了相应的修改


下面给出程序代码,大家可以从中体会一下显示列表的用法。

 

 

 

 #include <gl/glut.h>

#define WIDTH 400
#define HEIGHT 400

#include <math.h>
#define ColoredVertex(c, v) do{ glColor3fv(c); glVertex3fv(v); }while(0)

GLfloat angle = 0.0f;

void myDisplay(void)
{
    static int list = 0;
    if( list == 0 )
    {
        // 如果显示列表不存在,则创建
        /* GLfloat
            PointA[] = {-0.5, -5*sqrt(5)/48,  sqrt(3)/6},
            PointB[] = { 0.5, -5*sqrt(5)/48,  sqrt(3)/6},
            PointC[] = {   0, -5*sqrt(5)/48, -sqrt(3)/3},
            PointD[] = {   0, 11*sqrt(6)/48,          0}; */
        // 2007年4月27日修改
        GLfloat
            PointA[] = { 0.5f, -sqrt(6.0f)/12, -sqrt(3.0f)/6},
            PointB[] = {-0.5f, -sqrt(6.0f)/12, -sqrt(3.0f)/6},
            PointC[] = { 0.0f, -sqrt(6.0f)/12,  sqrt(3.0f)/3},
            PointD[] = { 0.0f,   sqrt(6.0f)/4,             0};
        GLfloat
            ColorR[] = {1, 0, 0},
            ColorG[] = {0, 1, 0},
            ColorB[] = {0, 0, 1},
            ColorY[] = {1, 1, 0};

        list = glGenLists(1);
        glNewList(list, GL_COMPILE);
        glBegin(GL_TRIANGLES);
        // 平面ABC
        ColoredVertex(ColorR, PointA);
        ColoredVertex(ColorG, PointB);
        ColoredVertex(ColorB, PointC);
        // 平面ACD
        ColoredVertex(ColorR, PointA);
        ColoredVertex(ColorB, PointC);
        ColoredVertex(ColorY, PointD);
        // 平面CBD
        ColoredVertex(ColorB, PointC);
        ColoredVertex(ColorG, PointB);
        ColoredVertex(ColorY, PointD);
        // 平面BAD
        ColoredVertex(ColorG, PointB);
        ColoredVertex(ColorR, PointA);
        ColoredVertex(ColorY, PointD);
        glEnd();
        glEndList();

        glEnable(GL_DEPTH_TEST);
    }
    // 已经创建了显示列表,在每次绘制正四面体时将调用它
    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
    glPushMatrix();
    glRotatef(angle, 1, 0.5, 0);
    glCallList(list);
    glPopMatrix();
    glutSwapBuffers();
}

void myIdle(void)
{
    ++angle;
    if( angle >= 360.0f )
        angle = 0.0f;
    myDisplay();
}

int main(int argc, char* argv[])
{
    glutInit(&argc, argv);
    glutInitDisplayMode(GLUT_RGBA | GLUT_DOUBLE);
    glutInitWindowPosition(200, 200);
    glutInitWindowSize(WIDTH, HEIGHT);
    glutCreateWindow("OpenGL 窗口");
    glutDisplayFunc(&myDisplay);
    glutIdleFunc(&myIdle);
    glutMainLoop();
    return 0;
}

 

 

 

在程序中,我们将绘制正四面体的OpenGL函数装到了一个显示列表中,但是,关于旋转的操作却在显示列表之外进行。这是因为如果把旋转的操作也装入到显示列表,则每次旋转的角度都是一样的,不会随着angle的值的变化而变化,于是就不能表现出动态的旋转效果了。
程序运行时,可能感觉到画面的立体感不足,这主要是因为没有使用光照的缘故。如果将glColor3fv函数去掉,改为设置各种材质,然后开启光照效果, 则可以产生更好的立体感。大家可以自己试着使用光照效果,唯一需要注意的地方就是法线向量的计算。由于这里的正四面体四个顶点坐标选取得比较特殊,使得正 四面体的中心坐标正好是(0, 0, 0),因此,每三个顶点坐标的平均值正好就是这三个顶点所组成的平面的法线向量的值。

 

 

void setNormal(GLfloat* Point1, GLfloat* Point2, GLfloat* Point3)
{
    GLfloat normal[3];
    int i;
    for(i=0; i<3; ++i)
        normal[i] = (Point1[i]+Point2[i]+Point3[i]) / 3;
    glNormal3fv(normal);
}

 

 

 

 


限于篇幅,这里就不给出完整的程序了。不过,大家可以自行尝试,看看使用光照后效果有何种改观。尤其是注意四面体各个表面交界的位置,在未使用光照前,几 乎看不清轮廓,在使用光照后,可比较容易的区分各个平面,因此立体感得到加强。(见图1,图2)当然了,这样的效果还不够。如果在各表面的交界处设置很多 细小的平面,进行平滑处理,则光照后的效果将更真实。但这已经远离本课的内容了。
http://blog.programfan.com/upfile/200703/20070303005337.jpg图一
http://blog.programfan.com/upfile/200703/20070303005342.jpg图二


小结
本课介绍了显示列表的知识和简单的应用。
可以把各种OpenGL函数调用的动作装到显示列表中,以后调用显示列表,就相当于调用了其中的OpenGL函数。显示列表中除了存放对OpenGL函数的调用外,不会存放其它内容。
使用显示列表的过程是:分配一个未使用的显示列表编号,把OpenGL函数调用装入显示列表,调用显示列表,销毁显示列表。
使用显示列表有可能带来程序运行速度的提升,但是这种提升并不一定会很明显。显示列表本身也存在一定的开销。
把绘制固定的物体的OpenGL函数放到一个显示列表中,是一种不错的编程思路。本课最后的例子中使用了这种思路。

 

转自http://blog.csdn.net/andyhuabing/article/details/6957264

分享到:
评论

相关推荐

    OpenGL入门学习之十一——纹理的使用入门.pdf

    ### OpenGL入门学习之十一——纹理的使用入门 #### 一、纹理的概念与基本操作 **纹理**是OpenGL中一种非常重要的特性,它允许开发者在3D模型表面贴上图像,以此来增强场景的真实感和细节表现力。纹理的使用不仅...

    net游戏编程源入门经典——C#篇

    通过《.NET游戏编程源入门经典——C#篇》的学习,读者可以逐步掌握游戏开发的核心技术,从零开始构建自己的游戏。配合代码示例,理论与实践相结合,将使学习过程更加高效。无论是想独立开发小游戏,还是希望在游戏...

    iPhone游戏开发入门经典——也适用于iPad

    《iPhone游戏开发入门经典——也适用于iPad》一书是由Peter Bakhirev、PJ Cabrera、Ian Marsh等多位在IT领域有着深厚经验的专家共同撰写的,旨在为初学者提供一套全面且实用的iPhone及iPad游戏开发指南。本书不仅...

    [.NET游戏编程入门经典—— C#篇]源文件

    通过阅读《.NET游戏编程入门经典——C#篇》的源文件,你可以逐步学习并实践这些知识点,从而开启你的游戏开发之旅。这些基础将为你打开通向更复杂游戏项目的大门,并为未来的进阶学习打下坚实基础。

    C# Nehe OpenGL第二课 绘制多边形

    在Nehe的这个第二课中,你将学习如何创建一个简单的OpenGL上下文,初始化必要的OpenGL状态,然后绘制一个多边形。首先,你需要设置视口、投影和模型视图矩阵。接着,定义多边形的顶点,然后调用`glBegin()`和`glEnd...

    OpenGL_Nehe.pdf

    ### OpenGL与NeHe教程知识点概览 #### 一、教程简介 - **目的与受众**:该教程旨在为初学者提供一个轻松简单的学习路径,帮助他们...无论是想要入门OpenGL还是希望深入掌握特定技术点的学习者,都可以从该教程中获益。

    OpenGL ES 3.x游戏开发 上卷 吴亚峰

    第七章讲述了3D模型的加载方式以及混合和雾效的基本原理与实现。 - **模型文件格式**:如OBJ、FBX等,这些格式包含了模型的顶点数据、纹理坐标等信息。 - **混合**:用于实现透明效果,例如半透明物体的表现。 - **...

    OpenGl 参考资料

    "OpenGL Super Bible"是学习OpenGL的权威书籍,包含大量示例代码和详细解释,适合深入学习。"OpenGL Red Book"则是官方文档,提供了规范和标准的详细说明。"C程序设计第5章课件与程序"可能是关于如何在C语言中使用...

    OpenGL编程指南

    1. **第七章:“混合、抗锯齿与雾效”**:这一章介绍创建逼真场景必不可少的技术——alpha混合(即透明度处理)、抗锯齿(减少边缘模糊)以及雾效。这些技术对于提升场景的真实感至关重要。 四、OpenGL编程实践 ...

    Android 4游戏编程入门经典

    ##### 第7章:OpenGLES简介 - **图形渲染基础**:介绍OpenGL ES的基本概念和技术。 - **绘制2D图形**:学习如何使用OpenGL ES进行2D图形渲染。 ##### 第8章:2D游戏编程技巧 - **碰撞检测算法**:讲解常见的碰撞...

    Android.游戏开发入门

    - **第7章:OpenGL ES入门**:提供了OpenGL ES的入门教程,适合初学者学习。 - **第8章:2D游戏编程技巧**:分享了一系列提高2D游戏质量的小技巧。 - **第10章:OpenGL ES:进入3D世界**:进一步介绍了OpenGL ES在3D...

    NeHe_OpenGL_PDF_NEW

    2. **绘制第一个多边形**:演示了如何使用OpenGL命令来绘制简单的2D图形。 3. **添加颜色**:讲解了如何为图形添加颜色,以及OpenGL的颜色模型。 4. **旋转**:介绍如何使用OpenGL的矩阵变换来进行旋转操作。 ...

    Three.js 入门指南(中文)

    - 想要学习 WebGL 但没有 OpenGL 背景的读者。 - 已经听说 Three.js,但缺乏完整教程的学习者。 - 对 Three.js 有一定了解,想要深入学习高级主题的开发者。 #### 四、内容组织结构 - **第 1 章**:介绍 Three....

    3D_case_study_using_opengl.pdf

    ### 第七章:结论——未来的可能性 - **总结与展望**:总结项目的成果,并探讨未来可能的研究方向和技术进步。 综上所述,这份3D案例研究涵盖了从基础图形绘制到高级光照和纹理映射等各个方面,为学习OpenGL提供了...

    Android 3D游戏开发技术宝典 源代码1

    3. **第7章 纹理映射**:纹理映射是为3D模型赋予颜色和细节的关键技术。它涉及到纹理坐标、纹理坐标映射、纹理合成和纹理过滤等,通过合适的纹理可以使3D模型更加生动逼真。 4. **第8章 3D基本形状**:本章讲解了3D...

Global site tag (gtag.js) - Google Analytics