青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

永遠也不完美的程序

不斷學習,不斷實踐,不斷的重構……

常用鏈接

統計

積分與排名

好友鏈接

最新評論

渲染狀態管理(轉)

來源:http://blog.csdn.net/duduliao/archive/2008/08/25/2827683.aspx
渲染狀態管理 出 處:Maple Studio
[ 2003-06-16 ] 作 者:maple


  目 錄
  1 基本思想
  2 實際問題
  3 渲染腳本


--------------------------------------------------------------------------------

  提高3D圖形程序的性能是個很大的課題。圖形程序的優化大致可以分成兩大任務,一是要有好的場景管理程序,能快速剔除不可見多邊形,并根據對象距相機遠近選擇合適的細節(LOD);二是要有好的渲染程序,能快速渲染送入渲染管線的可見多邊形。
  我們知道,使用OpenGL或Direct3D渲染圖形時,首先要設置渲染狀態,渲染狀態用于控制渲染器的渲染行為。應用程序可以通過改變渲染狀態來控制OpenGL或Direct3D的渲染行為。比如設置Vertex/Fragment Program、綁定紋理、打開深度測試、設置霧效等。
  改變渲染狀態對于顯卡而言是比較耗時的操作,而如果能合理管理渲染狀態,避免多余的狀態切換,將明顯提升圖形程序性能。這篇文章將討論渲染狀態的管理。

1、基本思想
  我們考慮一個典型的游戲場景,包含人、動物、植物、建筑、交通工具、武器等。稍微分析一下就會發現,實際上場景里很多對象的渲染狀態是一樣的,比如所有的人和動物的渲染狀態一般都一樣,所有的植物渲染狀態也一樣,同樣建筑、交通工具、武器也是如此。我們可以把具有相同的渲染狀態的對象歸為一組,然后分組渲染,對每組對象只需要在渲染前設置一次渲染狀態,并且還可以保存當前的渲染狀態,設置渲染狀態時只需改變和當前狀態不一樣的狀態。這樣可以大大減少多余的狀態切換。下面的代碼段演示了這種方法:

  // 渲染狀態組鏈表,由場景管理程序填充
  RenderStateGroupList groupList;
  // 當前渲染狀態
  RenderState curState;

  ……

  // 遍歷鏈表中的每個組
  RenderStateGroup *group = groupList.GetFirst();
  while ( group != NULL )
  {
    // 設置該組的渲染狀態
    RenderState *state = group->GetRenderState();
    state->ApplyRenderState( curState );

    // 該渲染狀態組的對象鏈表
    RenderableObjectList *objList = group->GetRenderableObjectList();
    // 遍歷對象鏈表的每個對象
    RenderableObject *obj = objList->GetFirst();
    while ( obj != NULL )
    {
      // 渲染對象
      obj->Render();
      obj = objList->GetNext();
    }

    group = groupList.GetNext();
  }

  其中RenderState類的ApplyRenderState方法形如:

  void RenderState::ApplyRenderState( RenderState &curState )
  {
    // 深度測試
    if ( depthTest != curState.depthTest )
    {
      SetDepthTest( depthTest );
      curState.depthTest = depthTest;
    }

    // Alpha測試
    if ( alphaTest != curState.alphaTest )
    {
      SetAlphaTest( alphaTest );
      curState.alphaTest = alphaTest;
    }

    // 其它渲染狀態
    ……
  }

  這些分組的渲染狀態一般被稱為Material或Shader。這里Material不同于OpenGL和Direct3D里面用于光照的材質, Shader也不同于OpenGL里面的Vertex/Fragment Program和Direct3D里面的Vertex/Pixel Shader。而是指封裝了的顯卡渲染圖形需要的狀態(也包括了OpenGL和Direct3D原來的Material和Shader)。
  從字面上看,Material(材質)更側重于對象表面外觀屬性的描述,而Shader(這個詞實在不好用中文表示)則有用程序控制對象表面外觀的含義。由于顯卡可編程管線的引入,渲染狀態中包含了Vertex/Fragment Program,這些小程序可以控制物體的渲染,所以我覺得將封裝的渲染狀態稱為Shader更合適。這篇文章也將稱之為Shader。
  上面的代碼段只是簡單的演示了渲染狀態管理的基本思路,實際上渲染狀態的管理需要考慮很多問題。

2、實際問題

  2.1 消耗時間問題
  改變渲染狀態時,不同的狀態消耗的時間并不一樣,甚至在不同條件下改變渲染狀態消耗的時間也不一樣。比如綁定紋理是一個很耗時的操作,而當紋理已經在顯卡的紋理緩存中時,速度就會非常快。而且隨著硬件和軟件的發展,一些很耗時的渲染狀態的消耗時間可能會有減少。因此并沒有一個準確的消耗時間的數據。
  雖然消耗時間無法量化,情況不同消耗的時間也不一樣,但一般來說下面這些狀態切換是比較消耗時間的:

Vertex/Fragment Program模式和固定管線模式的切換(FF,Fixed Function Pipeline)

Vertex/Fragment Program本身程序的切換。
改變Vertex/Fragment Program常量。
紋理切換。
頂點和索引緩存(Vertex & Index Buffers)切換。
  有時需要根據消耗時間的多少來做折衷,下面將會遇到這種情況。

  2.2 渲染狀態分類
  實際場景中,往往會出現這樣的情況,一類對象其它渲染狀態都一樣,只是紋理和頂點、索引數據不同。比如場景中的人,只是身材、長相、服裝等不同,也就是說只有紋理、頂點、索引數據不同,而其它如Vertex/Fragment Program、深度測試等渲染狀態都一樣。相反,一般不會存在紋理和頂點、索引數據相同,而其他渲染狀態不同的情況。我們可以把紋理、頂點、索引數據不歸入到Shader中,這樣場景中所有的人都可以用一個Shader來渲染,然后在這個Shader下對紋理進行分組排序,相同紋理的人放在一起渲染。

  2.3 多道渲染(Multipass Rendering)
  有些比較復雜的圖形效果,在低檔顯卡上需要渲染多次,每次渲染一種效果,然后用GL_BLEND合成為最終效果。這種方法叫多道渲染Multipass Rendering,渲染一次就是一個pass。比如做逐像素凹凸光照,需要計算環境光、漫射光凹凸效果、高光凹凸效果,在NV20顯卡上只需要1個 pass,而在NV10顯卡上則需要3個pass。Shader應該支持多道渲染,即一個Shader應該分別包含每個pass的渲染狀態。
  不同的pass往往渲染狀態和紋理都不同,而頂點、索引數據是一樣的。這帶來一個問題:是以對象為單位渲染,一次渲染一個對象的所有pass,然后渲染下一個對象;還是以pass為單位渲染,第一次渲染所有對象的第一個pass,第二次渲染所有對象的第二個pass。下面的程序段演示了這兩種方式:

  2.3.1 以對象為單位渲染

  // 渲染狀態組鏈表,由場景管理程序填充
  ShaderGroupList groupList;

  ……

  // 遍歷鏈表中的每個組
  ShaderGroup *group = groupList.GetFirst();
  while ( group != NULL )
  {
    Shader *shader = group->GetShader();
    RenderableObjectList *objList = group->GetRenderableObjectList();

    // 遍歷相同Shader的每個對象
    RenderableObject *obj = objList->GetFirst();
    while ( obj != NULL )
    {
      // 獲取shader的pass數
      int iNumPasses = shader->GetPassNum();
      for ( int i = 0; i < iNumPasses; i++ )
      {
        // 設置shader第i個pass的渲染狀態
        shader->ApplyPass( i );
        // 渲染對象
        obj->Render();
      }
      obj = objList->GetNext();
    }
    group = groupList->GetNext();
  }

  2.3.2 以pass為單位渲染

  // 渲染狀態組鏈表,由場景管理程序填充
  ShaderGroupList groupList;

  ……

  for ( int i = 0; i < MAX_PASSES_NUM; i++ )
  {
    // 遍歷鏈表中的每個組
    ShaderGroup *group = groupList.GetFirst();
    while ( group != NULL )
    {
      Shader *shader = group->GetShader();
      int iNumPasses = shader->GetPassNum();

      // 如果shader的pass數小于循環次數,跳過此shader
      if( i >= iNumPasses )
      {
        group = groupList->GetNext();
        continue;
      }

      // 設置shader第i個pass的渲染狀態
      shader->ApplyPass( i );
      RenderableObjectList *objList = group->GetRenderableObjectList();

      // 遍歷相同Shader的每個對象
      RenderableObject *obj = objList->GetFirst();
      while ( obj != NULL )
      {
        obj->Render();
        obj = objList->GetNext();
      }
      group = groupList->GetNext();
    }
  }

  這兩種方式各有什么優缺點呢?以對象為單位渲染,渲染一個對象的第一個pass后,馬上緊接著渲染這個對象的第二個pass,而每個pass的頂點和索引數據是相同的,因此第一個 pass將頂點和索引數據送入顯卡后,顯卡Cache中已經有了這個對象頂點和索引數據,后續pass不必重新將頂點和索引數據拷到顯卡,因此速度會非常快。而問題是每個pass的渲染狀態都不同,這使得實際上每次渲染都要設置新的渲染狀態,會產生大量的多余渲染狀態切換。
  以pass為單位渲染則正好相反,以Shader分組,相同Shader的對象一起渲染,可以只在這組開始時設置一次渲染狀態,相比以對象為單位,大大減少了渲染狀態切換。可是每次渲染的對象不同,因此每次都要將對象的頂點和索引數據拷貝到顯卡,會消耗不少時間。可見想減少渲染狀態切換就要頻繁拷貝頂點索引數據,而想減少拷貝頂點索引數據又不得不增加渲染狀態切換。魚與熊掌不可兼得。由于硬件條件和場景數據的情況比較復雜,具體哪種方法效率較高并沒有定式,兩種方法都有人使用,具體選用那種方法需要在實際環境測試后才能知道。

  2.3 多光源問題
  待續……

  2.4 陰影問題
  待續……

3、渲染腳本
  現在很多圖形程序都會自己定義一種腳本文件來描述Shader。
  比如較早的OGRE(Object-oriented Graphics Rendering Engine,面向對象圖形渲染引擎)的Material腳本,Quake3的Shader腳本,以及剛問世不久的Direct3D的Effect File,nVIDIA的CgFX腳本(文件格式與Direct3D Effect File兼容),ATI RenderMonkey使用的xml格式的腳本。OGRE Material和Quake3 Shader這兩種腳本比較有歷史了,不支持可編程渲染管線。而后面三種比較新的腳本都支持可編程渲染管線。

腳本 特性 范例
OGRE Material 封裝各種渲染狀態,不支持可編程渲染管線 >>>>
Quake3 Shader 封裝渲染狀態,支持一些特效,不支持可編程渲染管線 >>>>
Direct3D Effect File 封裝渲染狀態,支持multipass,支持可編程渲染管線 >>>>
nVIDIA CgFX腳本 封裝渲染狀態,支持multipass,支持可編程渲染管線 >>>>
ATI RenderMonkey腳本 封裝渲染狀態,支持multipass,支持可編程渲染管線 >>>>

  使用腳本來控制渲染有很多好處:

可以非常方便的修改一個物體的外觀而不需重新編寫或編譯程序。
可以用外圍工具以所見即所得的方式來創建、修改腳本文件(類似ATI RenderMonkey的工作方式),便于美工、關卡設計人員設定對象外觀,建立外圍工具與圖形引擎的聯系。
可以在渲染時將相同外觀屬性及渲染狀態的對象(也就是Shader相同的對象)歸為一組,然后分組渲染,對每組對象只需要在渲染前設置一次渲染狀態,大大減少了多余的狀態切換。


本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/duduliao/archive/2008/08/25/2827683.aspx

posted on 2009-06-13 11:44 狂爛球 閱讀(521) 評論(0)  編輯 收藏 引用 所屬分類: 圖形編程

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            国产精品二区二区三区| 亚洲日韩欧美一区二区在线| 国语自产精品视频在线看一大j8| 欧美日韩亚洲激情| 欧美视频一区二区| 国产精品老牛| 黑人巨大精品欧美黑白配亚洲| 国产热re99久久6国产精品| 国产亚洲精品久久久久久| 狠狠色噜噜狠狠色综合久| 亚洲国产毛片完整版| 亚洲精选国产| 亚洲一区久久久| 久久久人成影片一区二区三区观看| 欧美h视频在线| 一本色道久久综合狠狠躁篇怎么玩| 亚洲一区二区三区在线观看视频| 欧美在线观看一区二区| 欧美激情一区二区三区全黄 | 一区二区三区国产精华| 亚洲欧美日韩精品久久| 老鸭窝91久久精品色噜噜导演| 欧美日韩在线三级| 尤物99国产成人精品视频| 一本色道88久久加勒比精品| 久久天天躁夜夜躁狠狠躁2022| 最新日韩在线| 久久久xxx| 国产美女一区二区| 夜夜躁日日躁狠狠久久88av| 老牛影视一区二区三区| 亚洲影视在线播放| 欧美日韩一区二区三区四区在线观看| 国模精品娜娜一二三区| 亚洲视频二区| 亚洲高清成人| 久久五月激情| 国产在线欧美| 香蕉成人伊视频在线观看| 亚洲茄子视频| 免费成人在线视频网站| 国产一区二区激情| 亚洲欧美日韩精品久久| 国产香蕉久久精品综合网| 夜夜嗨av一区二区三区网站四季av | 亚洲一二三四久久| 亚洲国产精品成人久久综合一区 | 欧美激情国产日韩精品一区18| 国产欧美一区二区在线观看| 亚洲欧美日韩一区二区在线 | 欧美亚洲免费高清在线观看| 欧美日韩亚洲国产精品| av成人免费在线观看| 欧美激情亚洲自拍| 可以免费看不卡的av网站| 狠狠88综合久久久久综合网| 久久超碰97中文字幕| 亚洲欧美日韩天堂一区二区| 国产精品r级在线| 亚洲一区二区免费视频| 日韩亚洲在线| 国产精品地址| 欧美中文在线免费| 欧美在线免费视屏| 一色屋精品视频在线看| 欧美激情a∨在线视频播放| 久久综合给合| 日韩午夜精品| 亚洲午夜羞羞片| 国产一区二区三区黄视频| 看欧美日韩国产| 欧美sm视频| 一本色道久久综合亚洲精品不| 亚洲精品资源| 国产精品日韩久久久久| 亚洲福利一区| 在线午夜精品自拍| 国产一区二区日韩精品| 欧美激情中文不卡| 亚洲老板91色精品久久| 亚洲日本成人女熟在线观看| 欧美精品自拍| 欧美一级播放| 美女主播一区| 亚洲一区在线播放| 性感少妇一区| 夜夜嗨av一区二区三区网站四季av | 欧美~级网站不卡| 欧美精品麻豆| 欧美亚洲免费在线| 美女亚洲精品| 欧美日韩二区三区| 亚洲一区中文| 久久久国产成人精品| 日韩一区二区精品在线观看| 亚洲综合第一| 亚洲精品久久久久| 亚洲欧美综合一区| 日韩亚洲欧美一区二区三区| 欧美一区二区观看视频| 99re热这里只有精品免费视频| 午夜亚洲精品| 一本色道久久综合一区| 久久精品中文字幕一区| 在线视频亚洲| 久久全球大尺度高清视频| 亚洲主播在线| 欧美激情综合色综合啪啪| 久久久久久久久综合| 欧美日韩中文在线观看| 欧美好骚综合网| 黄色精品网站| 午夜在线观看免费一区| 亚洲午夜日本在线观看| 欧美电影免费观看| 欧美黄污视频| 亚洲大胆女人| 欧美中文字幕不卡| 久久av一区二区三区| 国产精品久久| 国产精品99久久久久久久久| 亚洲乱码一区二区| 欧美99在线视频观看| 蜜臀av性久久久久蜜臀aⅴ| 国产综合精品一区| 欧美一级欧美一级在线播放| 欧美亚洲尤物久久| 国产精品三区www17con| 亚洲直播在线一区| 久久av一区二区三区漫画| 国产视频一区在线观看| 欧美一区不卡| 你懂的国产精品| 亚洲人成毛片在线播放| 欧美精品乱人伦久久久久久 | 久久综合九色综合网站| 久久亚洲一区二区| 激情欧美一区二区三区| 久久亚洲综合色一区二区三区| 久久这里只有| 亚洲高清不卡| 欧美精品手机在线| 夜夜嗨av一区二区三区四区| 亚洲欧美一级二级三级| 国产精品一区二区久久精品| 欧美一级淫片播放口| 亚洲激情社区| 1024成人网色www| 欧美国产日韩视频| 亚洲精品一二| 亚洲永久精品国产| 国产欧美激情| 久久夜色精品国产欧美乱| 亚洲国产精品99久久久久久久久| 99视频+国产日韩欧美| 国产精品a级| 欧美中在线观看| 欧美激情a∨在线视频播放| 亚洲无玛一区| 久久亚洲春色中文字幕| 欧美激情精品久久久久久大尺度| 亚洲观看高清完整版在线观看| 欧美成在线观看| 亚洲一级在线观看| 免费不卡在线观看av| 一区二区三区四区国产精品| 国产嫩草影院久久久久| 免费成人高清视频| 亚洲欧美日本日韩| 亚洲国产成人在线播放| 亚洲欧美国产精品va在线观看| 狠狠色综合播放一区二区| 欧美精品在线观看一区二区| 午夜精品一区二区三区在线播放| 欧美高清一区二区| 性色av一区二区三区在线观看| 在线成人性视频| 国产精品美女主播| 欧美aⅴ一区二区三区视频| 亚洲欧洲av一区二区三区久久| 亚洲国产美女精品久久久久∴| 久久国产免费| 亚洲视频综合| 亚洲毛片在线| 亚洲国产日日夜夜| 国产亚洲永久域名| 欧美日韩欧美一区二区| 久热这里只精品99re8久| 亚洲欧美日韩在线观看a三区| 亚洲国产精品久久久| 久久一区二区三区四区五区| 欧美一区二区三区免费大片| 一区二区欧美在线观看| 亚洲国产精品电影| 伊人久久亚洲美女图片| 国产视频在线观看一区二区| 国产精品高潮呻吟久久av无限| 欧美福利影院| 欧美88av| 蜜桃久久精品乱码一区二区|