• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            OpenGL顯示列表

              目 錄
              16.1 顯示列表概論
              16.2 創(chuàng)建和執(zhí)行顯示列表
              16.3 管理顯示列表
              16.4 多級顯示列表

             

              OpenGL顯示列表(Display List)是由一組預先存儲起來的留待以后調用的OpenGL函數(shù)語句組成的,當調用這張顯示列表時就依次執(zhí)行表中所列出的函數(shù)語句。前面內容所舉出的例子都是瞬時給出函數(shù)命令,則OpenGL瞬時執(zhí)行相應的命令,這種繪圖方式叫做立即或瞬時方式(immediate mode)。本章將詳細地講述顯示列表的基本概論、創(chuàng)建、執(zhí)行、管理以及多級顯示列表的應用等內容。

            16.1、顯示列表概論

              16.1.1 顯示列表的優(yōu)勢
              OpenGL顯示列表的設計能優(yōu)化程序運行性能,尤其是網絡性能。它被設計成命令高速緩存,而不是動態(tài)數(shù)據(jù)庫緩存。也就是說,一旦建立了顯示列表,就不能修改它。因為若顯示列表可以被修改,則顯示列表的搜索、內存管理的執(zhí)行等開銷會降低性能。
              采用顯示列表方式繪圖一般要比瞬時方式快,尤其是顯示列表方式可以大量地提高網絡性能,即當通過網絡發(fā)出繪圖命令時,由于顯示列表駐留在服務器中,因而使網絡的負擔減輕到最小。另外,在單用戶的機器上,顯示列表同樣可以提高效率。因為一旦顯示列表被處理成適合于圖形硬件的格式,則不同的OpenGL實現(xiàn)對命令的優(yōu)化程度也不同。例如旋轉矩陣函數(shù)glRotate*(),若將它置于顯示列表中,則可大大提高性能。因為旋轉矩陣的計算并不簡單,包含有平方、三角函數(shù)等復雜運算,而在顯示列表中,它只被存儲為最終的旋轉矩陣,于是執(zhí)行起來如同硬件執(zhí)行函數(shù)glMultMatrix()一樣快。一般來說,顯示列表能將許多相鄰的矩陣變換結合成單個的矩陣乘法,從而加快速度。

              16.1.2 顯示列表的適用場合
              并不是只要調用顯示列表就能優(yōu)化程序性能。因為調用顯示列表本身時程序也有一些開銷,若一個顯示列表太小,這個開銷將超過顯示列表的優(yōu)越性。下面給出顯示列表能最大優(yōu)化的場合:
             
            • 矩陣操作
              大部分矩陣操作需要OpenGL計算逆矩陣,矩陣及其逆矩陣都可以保存在顯示列表中。

               
            • 光柵位圖和圖像
              程序定義的光柵數(shù)據(jù)不一定是適合硬件處理的理想格式。當編譯組織一個顯示列表時,OpenGL可能把數(shù)據(jù)轉換成硬件能夠接受的數(shù)據(jù),這可以有效地提高畫位圖的速度。

               
            • 光、材質和光照模型
              當用一個比較復雜的光照環(huán)境繪制場景時,可以為場景中的每個物體改變材質。但是材質計算較多,因此設置材質可能比較慢。若把材質定義放在顯示列表中,則每次改換材質時就不必重新計算了。因為計算結果存儲在表中,因此能更快地繪制光照場景。

               
            • 紋理
              因為硬件的紋理格式可能與OpenGL格式不一致,若把紋理定義放在顯示列表中,則在編譯顯示列表時就能對格式進行轉換,而不是在執(zhí)行中進行,這樣就能大大提高效率。

               
            • 多邊形的圖案填充模式
              即可將定義的圖案放在顯示列表中。
            16.2、創(chuàng)建和執(zhí)行顯示列表

              16.2.1 創(chuàng)建顯示列表
              OpenGL提供類似于繪制圖元的結構即glBegin()與glEnd()的形式創(chuàng)建顯示列表,其相應的函數(shù)為:

             
            void glNewList(GLuint list,GLenum mode);

              說明一個顯示列表的開始,其后的OpenGL函數(shù)存入顯示列表中,直至調用結束表的函數(shù)(見下面)。參數(shù)list是一個正整數(shù),它標志唯一的顯示列表。參數(shù)mode的可能值有GL_COMPILE和GL_COMPILE_AND_EXECUTE。若要使后面的函數(shù)語句只存入而不執(zhí)行,則用GL_COMPILE;若要使后面的函數(shù)語句存入表中且按瞬時方式執(zhí)行一次,則用GL_COMPILE_AND_EXECUTE。
             

            void glEndList(void);

              標志顯示列表的結束。
              注意:并不是所有的OpenGL函數(shù)都可以在顯示列表中存儲且通過顯示列表執(zhí)行。一般來說,用于傳遞參數(shù)或返回數(shù)值的函數(shù)語句不能存入顯示列表,因為這張表有可能在參數(shù)的作用域之外被調用;如果在定義顯示列表時調用了這樣的函數(shù),則它們將按瞬時方式執(zhí)行并且不保存在顯示列表中,有時在調用執(zhí)行顯示列表函數(shù)時會產生錯誤。以下列出的是不能存入顯示列表的OpenGL函數(shù):
             

             
            glDeleteLists()  glIsEnable()
            glFeedbackBuffer()  glIsList()
            glFinish()  glPixelStore()
            glGenLists()  glRenderMode()
            glGet*()  glSelectBuffer()

              16.2.2 執(zhí)行顯示列表
              在建立顯示列表以后就可以調用執(zhí)行顯示列表的函數(shù)來執(zhí)行它,并且允許在程序中多次執(zhí)行同一顯示列表,同時也可以與其它函數(shù)的瞬時方式混合使用。顯示列表執(zhí)行的函數(shù)形式如下:

             

            void glCallList(GLuint list);

              執(zhí)行顯示列表。參數(shù)list指定被執(zhí)行的顯示列表。顯示列表中的函數(shù)語句按它們被存放的順序依次執(zhí)行;若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、管理顯示列表
              在上一節(jié)例子中,我們使用了一個正整數(shù)作為顯示列表的索引。但是在實際應用中,一般不采用這種方式,尤其在創(chuàng)建多個顯示列表的情況下。如果這樣做,則有可能選用某個正在被占用的索引,并且覆蓋這個已經存在的顯示列表,對程序運行造成危害。為了避免意外刪除,可以調用函數(shù)glGenList()來產生一個沒有用過的顯示列表,或調用glIsList()來決定是否指定的顯示列表被占用。此外,在管理顯示列表的過程中,還可調用函數(shù)glDeleteLists()來刪除一個或一個范圍內的顯示列表。下面分別介紹這些函數(shù):
            GLuint glGenList(GLsizei range);

              分配range個相鄰的未被占用的顯示列表索引。這個函數(shù)返回的是一個正整數(shù)索引值,它是一組連續(xù)空索引的第一個值。返回的索引都標志為空且已被占用,以后再調用這個函數(shù)時不再返回這些索引。若申請索引的指定數(shù)目不能滿足或range為0則函數(shù)返回0。
             

            GLboolean glIsList(GLuint list);

             詢問顯示列表是否已被占用的情況。若索引list已被占用,則函數(shù)返回TURE;反之,返回FAULSE。

            void glDeleteLists(GLuint list,GLsizei range);

              刪除一組連續(xù)的顯示列表,即從參數(shù)list所指示的顯示列表開始,刪除range個顯示列表,并且刪除后的這些索引重新有效。若刪除一個沒有建立的顯示列表則忽略刪除操作。
              當建立一個與已經存在的顯示列表索引相同的顯示列表時,OpenGL將自動刪除舊表。這一節(jié)舉個例子來說,如果將上一節(jié)程序/***.c*/中所創(chuàng)建的顯示列表改為以下代碼:
             

             
            listIndex=glGenLists(1);
            if(listIndex!=0)
            {
              glNewList(listIndex,GL_COMPILE);
              ...
              glEndList();
            }

              那么,這個程序將更加優(yōu)化實用。讀者自己不妨試試,同時還可用它多創(chuàng)建幾個顯示列表,或者再刪除一個,看看效果怎樣?

            16.4、多級顯示列表
              多級顯示列表的建立就是在一個顯示列表中調用另一個顯示列表,也就是說,在函數(shù)glNewList()與glEndList()之間調用glCallList()。多級顯示列表對于構造由多個元件組成的物體十分有用,尤其是某些元件需要重復使用的情況。但為了避免無窮遞歸,顯示列表的嵌套深度最大為64(也許更高些,這依賴于不同的OpenGL實現(xiàn)),當然也可調用函數(shù)glGetIntegerv()來獲得這個最大嵌套深度值。
              OpenGL在建立的顯示列表中允許調用尚未建立的表,當?shù)谝粋€顯示列表調用第二個并沒 定義的表時,不會發(fā)生任何操作。另外,也允許用一個顯示列表包含幾個低級的顯示列表來模擬建立一個可編輯的顯示列表。如下一段代碼:

             

            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);要編輯頂點,只需重新建立相應的該頂點顯示列表。

            posted on 2006-01-17 01:10 zmj 閱讀(5667) 評論(6)  編輯 收藏 引用

            評論

            # re: OpenGL顯示列表 2009-01-01 10:19 11

            寫的很好的,對于用時查閱很方便。  回復  更多評論   

            # re: OpenGL顯示列表 2010-11-24 13:45 wql

            怎么創(chuàng)建多個顯示列表啊?
            我有一道有關這的題目
            讓一個桌子。茶壺和兔子 組合顯示  回復  更多評論   

            # re: OpenGL顯示列表 2010-11-24 13:46 wql

            能不能幫我解決下  回復  更多評論   

            # re: OpenGL顯示列表 2010-11-24 13:47 wql

            我的郵箱wqiulin@163.com 謝謝  回復  更多評論   

            # re: OpenGL顯示列表 2012-10-18 13:53 陳志遠

            非常感謝  回復  更多評論   

            # re: OpenGL顯示列表 2013-02-21 17:43 葉盧慶

            真奇怪,你的博客可以開放評論  回復  更多評論   

            亚洲综合久久夜AV | 久久精品成人| 国产精品久久久久aaaa| 2020最新久久久视精品爱| 精品无码人妻久久久久久| 亚洲欧美精品一区久久中文字幕 | 国产精品久久99| 一级做a爰片久久毛片16| 亚洲精品无码久久毛片| 国产V综合V亚洲欧美久久| 色99久久久久高潮综合影院| 国产精品一区二区久久不卡| 欧美伊人久久大香线蕉综合69| 久久天天躁狠狠躁夜夜网站 | 很黄很污的网站久久mimi色| 午夜精品久久久内射近拍高清| 久久久一本精品99久久精品66 | 蜜臀av性久久久久蜜臀aⅴ| 精品久久综合1区2区3区激情 | 国产精品99久久久久久猫咪| 人妻无码αv中文字幕久久琪琪布| 国产精品99久久久久久猫咪| 精品国产一区二区三区久久久狼 | 欧美伊人久久大香线蕉综合69| 国产精品福利一区二区久久| 中文字幕久久精品无码| 亚洲а∨天堂久久精品| 久久久久99精品成人片| 91精品久久久久久无码| 99久久99这里只有免费费精品| 丁香色欲久久久久久综合网| 亚洲国产精品综合久久一线| 久久久久久亚洲精品不卡| 精品久久久久久无码人妻热| 久久91精品国产91久久麻豆 | 久久综合给合综合久久| 亚洲狠狠久久综合一区77777| 久久se精品一区精品二区| 精品九九久久国内精品| 久久99精品国产麻豆宅宅| 欧美精品一区二区精品久久|