目 錄 16.1
顯示列表概論 16.2
創建和執行顯示列表 16.3
管理顯示列表 16.4
多級顯示列表
OpenGL顯示列表(
Display List)是由一組預先存儲起來的留待以后調用的OpenGL函數語句組成的,當調用這張顯示列表時就依次執行表中所列出的函數語句。前面內容所舉出的例子都是瞬時給出函數命令,則OpenGL瞬時執行相應的命令,這種繪圖方式叫做立即或瞬時方式(
immediate mode)。本章將詳細地講述顯示列表的基本概論、創建、執行、管理以及多級顯示列表的應用等內容。
16.1、顯示列表概論 16.1.1 顯示列表的優勢 OpenGL顯示列表的設計能優化程序運行性能,尤其是網絡性能。它被設計成命令高速緩存,而不是動態數據庫緩存。也就是說,一旦建立了顯示列表,就不能修改它。因為若顯示列表可以被修改,則顯示列表的搜索、內存管理的執行等開銷會降低性能。
采用顯示列表方式繪圖一般要比瞬時方式快,尤其是顯示列表方式可以大量地提高網絡性能,即當通過網絡發出繪圖命令時,由于顯示列表駐留在服務器中,因而使網絡的負擔減輕到最小。另外,在單用戶的機器上,顯示列表同樣可以提高效率。因為一旦顯示列表被處理成適合于圖形硬件的格式,則不同的OpenGL實現對命令的優化程度也不同。例如旋轉矩陣函數glRotate*(),若將它置于顯示列表中,則可大大提高性能。因為旋轉矩陣的計算并不簡單,包含有平方、三角函數等復雜運算,而在顯示列表中,它只被存儲為最終的旋轉矩陣,于是執行起來如同硬件執行函數glMultMatrix()一樣快。一般來說,顯示列表能將許多相鄰的矩陣變換結合成單個的矩陣乘法,從而加快速度。
16.1.2 顯示列表的適用場合 并不是只要調用顯示列表就能優化程序性能。因為調用顯示列表本身時程序也有一些開銷,若一個顯示列表太小,這個開銷將超過顯示列表的優越性。下面給出顯示列表能最大優化的場合:
- 矩陣操作
大部分矩陣操作需要OpenGL計算逆矩陣,矩陣及其逆矩陣都可以保存在顯示列表中。
- 光柵位圖和圖像
程序定義的光柵數據不一定是適合硬件處理的理想格式。當編譯組織一個顯示列表時,OpenGL可能把數據轉換成硬件能夠接受的數據,這可以有效地提高畫位圖的速度。
- 光、材質和光照模型
當用一個比較復雜的光照環境繪制場景時,可以為場景中的每個物體改變材質。但是材質計算較多,因此設置材質可能比較慢。若把材質定義放在顯示列表中,則每次改換材質時就不必重新計算了。因為計算結果存儲在表中,因此能更快地繪制光照場景。
- 紋理
因為硬件的紋理格式可能與OpenGL格式不一致,若把紋理定義放在顯示列表中,則在編譯顯示列表時就能對格式進行轉換,而不是在執行中進行,這樣就能大大提高效率。
- 多邊形的圖案填充模式
即可將定義的圖案放在顯示列表中。
16.2、創建和執行顯示列表 16.2.1 創建顯示列表 OpenGL提供類似于繪制圖元的結構即glBegin()與glEnd()的形式創建顯示列表,其相應的函數為:
void glNewList(GLuint list,GLenum mode);
說明一個顯示列表的開始,其后的OpenGL函數存入顯示列表中,直至調用結束表的函數(見下面)。參數list是一個正整數,它標志唯一的顯示列表。參數mode的可能值有GL_COMPILE和GL_COMPILE_AND_EXECUTE。若要使后面的函數語句只存入而不執行,則用GL_COMPILE;若要使后面的函數語句存入表中且按瞬時方式執行一次,則用GL_COMPILE_AND_EXECUTE。
void glEndList(void);
標志顯示列表的結束。
注意:并不是所有的OpenGL函數都可以在顯示列表中存儲且通過顯示列表執行。一般來說,用于傳遞參數或返回數值的函數語句不能存入顯示列表,因為這張表有可能在參數的作用域之外被調用;如果在定義顯示列表時調用了這樣的函數,則它們將按瞬時方式執行并且不保存在顯示列表中,有時在調用執行顯示列表函數時會產生錯誤。以下列出的是不能存入顯示列表的OpenGL函數:
glDeleteLists() glIsEnable()
glFeedbackBuffer() glIsList()
glFinish() glPixelStore()
glGenLists() glRenderMode()
glGet*() glSelectBuffer()
16.2.2 執行顯示列表
在建立顯示列表以后就可以調用執行顯示列表的函數來執行它,并且允許在程序中多次執行同一顯示列表,同時也可以與其它函數的瞬時方式混合使用。顯示列表執行的函數形式如下:
void glCallList(GLuint list);
執行顯示列表。參數list指定被執行的顯示列表。顯示列表中的函數語句按它們被存放的順序依次執行;若list沒有定義,則不會產生任何事情。下面舉出一個應用顯示列表的簡單例子:
例16-1 顯示列表例程(displist.c)
#include "glos.h"
#include <GL/gl.h>
#include <GL/glu.h>
#include <GL/glaux.h>
void myinit(void);
void drawLine(void);
void CALLBACK display(void);
void CALLBACK myReshape(GLsizei w, GLsizei h);
GLuint listName = 1;
void myinit (void)
{
glNewList (listName, GL_COMPILE);
glColor3f (1.0, 0.0, 0.0);
glBegin (GL_TRIANGLES);
glVertex2f (0.0, 0.0);
glVertex2f (1.0, 0.0);
glVertex2f (0.0, 1.0);
glEnd ();
glTranslatef (1.5, 0.0, 0.0);
glEndList ();
glShadeModel (GL_FLAT);
}
void drawLine (void)
{
glColor3f(1.0,1.0,0.0);
glBegin (GL_LINES);
glVertex2f (0.0, 0.5);
glVertex2f (5.0, 0.5);
glEnd ();
}
void CALLBACK display(void)
{
GLuint i;
glClear (GL_COLOR_BUFFER_BIT);
glColor3f (0.0, 1.0, 0.0);
glPushMatrix();
for (i = 0; i <5; i++)
glCallList (listName);
drawLine ();
glPopMatrix();
glFlush ();
}
void CALLBACK myReshape(GLsizei w, GLsizei h)
{
glViewport(0, 0, w, h);
glMatrixMode(GL_PROJECTION);
glLoadIdentity();
if (w <= h)
gluOrtho2D (0.0, 2.0, -0.5 * (GLfloat) h/(GLfloat) w, 1.5 * (GLfloat) h/(GLfloat) w);
else
gluOrtho2D (0.0, 2.0 * (GLfloat) w/(GLfloat) h, -0.5, 1.5); glMatrixMode(GL_MODELVIEW);
glLoadIdentity();
}
void main(void)
{
auxInitDisplayMode (AUX_SINGLE | AUX_RGBA);
auxInitPosition (10, 200, 400, 50);
auxInitWindow ("Display List");
myinit ();
auxReshapeFunc (myReshape);
auxMainLoop(display);
}
以上程序運行結果是顯示五個顯示列表中定義的紅色三角形,然后再繪制一條非表中的黃色線段。
 |
圖16-1 顯示列表 |
16.3、管理顯示列表
在上一節例子中,我們使用了一個正整數作為顯示列表的索引。但是在實際應用中,一般不采用這種方式,尤其在創建多個顯示列表的情況下。如果這樣做,則有可能選用某個正在被占用的索引,并且覆蓋這個已經存在的顯示列表,對程序運行造成危害。為了避免意外刪除,可以調用函數glGenList()來產生一個沒有用過的顯示列表,或調用glIsList()來決定是否指定的顯示列表被占用。此外,在管理顯示列表的過程中,還可調用函數glDeleteLists()來刪除一個或一個范圍內的顯示列表。下面分別介紹這些函數: GLuint glGenList(GLsizei range);
分配range個相鄰的未被占用的顯示列表索引。這個函數返回的是一個正整數索引值,它是一組連續空索引的第一個值。返回的索引都標志為空且已被占用,以后再調用這個函數時不再返回這些索引。若申請索引的指定數目不能滿足或range為0則函數返回0。
GLboolean glIsList(GLuint list);
詢問顯示列表是否已被占用的情況。若索引list已被占用,則函數返回TURE;反之,返回FAULSE。
void glDeleteLists(GLuint list,GLsizei range);
刪除一組連續的顯示列表,即從參數list所指示的顯示列表開始,刪除range個顯示列表,并且刪除后的這些索引重新有效。若刪除一個沒有建立的顯示列表則忽略刪除操作。
當建立一個與已經存在的顯示列表索引相同的顯示列表時,OpenGL將自動刪除舊表。這一節舉個例子來說,如果將上一節程序/***.c*/中所創建的顯示列表改為以下代碼:
listIndex=glGenLists(1);
if(listIndex!=0)
{
glNewList(listIndex,GL_COMPILE);
...
glEndList();
}
那么,這個程序將更加優化實用。讀者自己不妨試試,同時還可用它多創建幾個顯示列表,或者再刪除一個,看看效果怎樣?
16.4、多級顯示列表
多級顯示列表的建立就是在一個顯示列表中調用另一個顯示列表,也就是說,在函數glNewList()與glEndList()之間調用glCallList()。多級顯示列表對于構造由多個元件組成的物體十分有用,尤其是某些元件需要重復使用的情況。但為了避免無窮遞歸,顯示列表的嵌套深度最大為64(也許更高些,這依賴于不同的OpenGL實現),當然也可調用函數glGetIntegerv()來獲得這個最大嵌套深度值。
OpenGL在建立的顯示列表中允許調用尚未建立的表,當第一個顯示列表調用第二個并沒 定義的表時,不會發生任何操作。另外,也允許用一個顯示列表包含幾個低級的顯示列表來模擬建立一個可編輯的顯示列表。如下一段代碼:
glNewList(1,GL_COMPILE);
glVertex3fv(v1);
glEndList();
glNewList(2,GL_COMPILE);
glVertex3fv(v2);
glEndList();
glNewList(3,GL_COMPILE);
glVertex3fv(v3);
glEndList();
glNewList(4,GL_COMPILE);
glBegin(GL_POLYGON);
glCallList(1);
glCallList(2);
glCallList(3);
glEnd();
glEndList();
這樣,要繪制三角形就可以調用顯示列表4了,即調用glCallList(4);要編輯頂點,只需重新建立相應的該頂點顯示列表。