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

李錦俊(mybios)的blog

游戲開發(fā) C++ Cocos2d-x OpenGL DirectX 數(shù)學(xué) 計(jì)算機(jī)圖形學(xué) SQL Server

  C++博客 :: 首頁 :: 聯(lián)系 :: 聚合  :: 管理
  86 Posts :: 0 Stories :: 370 Comments :: 0 Trackbacks

公告

QQ:30743734
EMain:mybios@qq.com

常用鏈接

留言簿(16)

我參與的團(tuán)隊(duì)

最新隨筆

搜索

  •  

積分與排名

  • 積分 - 373919
  • 排名 - 67

最新評(píng)論

閱讀排行榜

評(píng)論排行榜

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

文檔目錄:??
  基本思想??
  實(shí)際問題??
  渲染腳本??

文檔內(nèi)容:??

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

// 渲染狀態(tài)組鏈表,由場(chǎng)景管理程序填充??
RenderStateGroupList groupList;??
// 當(dāng)前渲染狀態(tài)??
RenderState curState;??

……??

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

? ???// 該渲染狀態(tài)組的對(duì)象鏈表??
? ???RenderableObjectList *objList = group->GetRenderableObjectList();??
? ???// 遍歷對(duì)象鏈表的每個(gè)對(duì)象??
? ???RenderableObject *obj = objList->GetFirst();??
? ???while ( obj != NULL )??
? ???{??
? ?? ?? ?// 渲染對(duì)象??
? ?? ?? ?obj->Render();??

? ?? ?? ?obj = objList->GetNext();??
? ???}??

? ???group = groupList.GetNext();? ?
}??
? ?
//其中RenderState類的ApplyRenderState方法形如:? ?
void RenderState::ApplyRenderState( RenderState &curState )? ?
{??
? ???// 深度測(cè)試? ?
? ???if ( depthTest != curState.depthTest )??
? ???{??
? ?? ?? ?SetDepthTest( depthTest );??
? ?? ?? ?curState.depthTest = depthTest;??
? ???}??

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

? ???// 其它渲染狀態(tài)??
? ???……??
}? ?? ?


  這些分組的渲染狀態(tài)一般被稱為Material或Shader。這里Material不同于OpenGL和Direct3D里面用于光照的材質(zhì),Shader也不同于OpenGL里面的Vertex/Fragment Program和Direct3D里面的Vertex/Pixel Shader。而是指封裝了的顯卡渲染圖形需要的狀態(tài)(也包括了OpenGL和Direct3D原來的Material和Shader)。??

  從字面上看,Material(材質(zhì))更側(cè)重于對(duì)象表面外觀屬性的描述,而Shader(這個(gè)詞實(shí)在不好用中文表示)則有用程序控制對(duì)象表面外觀的含義。由于顯卡可編程管線的引入,渲染狀態(tài)中包含了Vertex/Fragment Program,這些小程序可以控制物體的渲染,所以我覺得將封裝的渲染狀態(tài)稱為Shader更合適。這篇文章也將稱之為Shader。??

  上面的代碼段只是簡(jiǎn)單的演示了渲染狀態(tài)管理的基本思路,實(shí)際上渲染狀態(tài)的管理需要考慮很多問題。??
渲染狀態(tài)管理的問題??
 ??

 消耗時(shí)間問題??
  改變渲染狀態(tài)時(shí),不同的狀態(tài)消耗的時(shí)間并不一樣,甚至在不同條件下改變渲染狀態(tài)消耗的時(shí)間也不一樣。比如綁定紋理是一個(gè)很耗時(shí)的操作,而當(dāng)紋理已經(jīng)在顯卡的紋理緩存中時(shí),速度就會(huì)非常快。而且隨著硬件和軟件的發(fā)展,一些很耗時(shí)的渲染狀態(tài)的消耗時(shí)間可能會(huì)有減少。因此并沒有一個(gè)準(zhǔn)確的消耗時(shí)間的數(shù)據(jù)。??

  雖然消耗時(shí)間無法量化,情況不同消耗的時(shí)間也不一樣,但一般來說下面這些狀態(tài)切換是比較消耗時(shí)間的:??

Vertex/Fragment Program模式和固定管線模式的切換(FF,F(xiàn)ixed Function Pipeline)? ?

Vertex/Fragment Program本身程序的切換? ?

改變Vertex/Fragment Program常量? ?

紋理切換? ?

頂點(diǎn)和索引緩存(Vertex & Index Buffers)切換? ?

  有時(shí)需要根據(jù)消耗時(shí)間的多少來做折衷,下面將會(huì)遇到這種情況。? ?

? ?

 渲染狀態(tài)分類??
  實(shí)際場(chǎng)景中,往往會(huì)出現(xiàn)這樣的情況,一類對(duì)象其它渲染狀態(tài)都一樣,只是紋理和頂點(diǎn)、索引數(shù)據(jù)不同。比如場(chǎng)景中的人,只是身材、長(zhǎng)相、服裝等不同,也就是說只有紋理、頂點(diǎn)、索引數(shù)據(jù)不同,而其它如Vertex/Fragment Program、深度測(cè)試等渲染狀態(tài)都一樣。相反,一般不會(huì)存在紋理和頂點(diǎn)、索引數(shù)據(jù)相同,而其他渲染狀態(tài)不同的情況。我們可以把紋理、頂點(diǎn)、索引數(shù)據(jù)不歸入到Shader中,這樣場(chǎng)景中所有的人都可以用一個(gè)Shader來渲染,然后在這個(gè)Shader下對(duì)紋理進(jìn)行分組排序,相同紋理的人放在一起渲染。??
 多道渲染(Multipass Rendering)??
  有些比較復(fù)雜的圖形效果,在低檔顯卡上需要渲染多次,每次渲染一種效果,然后用GL_BLEND合成為最終效果。這種方法叫多道渲染Multipass Rendering,渲染一次就是一個(gè)pass。比如做逐像素凹凸光照,需要計(jì)算環(huán)境光、漫射光凹凸效果、高光凹凸效果,在NV20顯卡上只需要1個(gè)pass,而在NV10顯卡上則需要3個(gè)pass。Shader應(yīng)該支持多道渲染,即一個(gè)Shader應(yīng)該分別包含每個(gè)pass的渲染狀態(tài)。??

? ? 不同的pass往往渲染狀態(tài)和紋理都不同,而頂點(diǎn)、索引數(shù)據(jù)是一樣的。這帶來一個(gè)問題:是以對(duì)象為單位渲染,一次渲染一個(gè)對(duì)象的所有pass,然后渲染下一個(gè)對(duì)象;還是以pass為單位渲染,第一次渲染所有對(duì)象的第一個(gè)pass,第二次渲染所有對(duì)象的第二個(gè)pass。下面的程序段演示了這兩種方式:??

??以對(duì)象為單位渲染? ?

// 渲染狀態(tài)組鏈表,由場(chǎng)景管理程序填充??
ShaderGroupList groupList;??

……??

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

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

? ?? ?? ?obj = objList->GetNext();??
? ???}??
? ?
? ???group = groupList->GetNext();??
}??
? ???

以pass為單位渲染? ?
? ?
// 渲染狀態(tài)組鏈表,由場(chǎng)景管理程序填充??
ShaderGroupList groupList;??
? ?
……??
? ?
for ( int i = 0; i < MAX_PASSES_NUM; i++ )
{
// 遍歷鏈表中的每個(gè)組
ShaderGroup *group = groupList.GetFirst();
while ( group != NULL )
{
Shader *shader = group->GetShader();??
? ?? ?? ?int iNumPasses = shader->GetPassNum();??
? ?? ?? ?// 如果shader的pass數(shù)小于循環(huán)次數(shù),跳過此shader??
? ?? ?? ?if( i >= iNumPasses )??
? ?? ?? ?{??
? ?? ?? ?? ? group = groupList->GetNext();??
? ?? ?? ?? ? continue;??
? ?? ?? ?}??

? ?? ?? ?// 設(shè)置shader第i個(gè)pass的渲染狀態(tài)??
? ?? ?? ?shader->ApplyPass( i );??

? ?? ?? ?RenderableObjectList *objList =? ?
? ?? ?? ?? ? group->GetRenderableObjectList();??
? ?
? ?? ?? ?// 遍歷相同Shader的每個(gè)對(duì)象??
? ?? ?? ?RenderableObject *obj = objList->GetFirst();??
? ?? ?? ?while ( obj != NULL )??
? ?? ?? ?{??
? ?? ?? ?? ? obj->Render();??

? ?? ?? ?? ? obj = objList->GetNext();??
? ?? ?? ?}??

? ?? ?? ?group = groupList->GetNext();??
? ???}??
}??
? ?

? ?? ?
  這兩種方式各有什么優(yōu)缺點(diǎn)呢???

  以對(duì)象為單位渲染,渲染一個(gè)對(duì)象的第一個(gè)pass后,馬上緊接著渲染這個(gè)對(duì)象的第二個(gè)pass,而每個(gè)pass的頂點(diǎn)和索引數(shù)據(jù)是相同的,因此第一個(gè)pass將頂點(diǎn)和索引數(shù)據(jù)送入顯卡后,顯卡Cache中已經(jīng)有了這個(gè)對(duì)象頂點(diǎn)和索引數(shù)據(jù),后續(xù)pass不必重新將頂點(diǎn)和索引數(shù)據(jù)拷到顯卡,因此速度會(huì)非常快。而問題是每個(gè)pass的渲染狀態(tài)都不同,這使得實(shí)際上每次渲染都要設(shè)置新的渲染狀態(tài),會(huì)產(chǎn)生大量的多余渲染狀態(tài)切換。??

  以pass為單位渲染則正好相反,以Shader分組,相同Shader的對(duì)象一起渲染,可以只在這組開始時(shí)設(shè)置一次渲染狀態(tài),相比以對(duì)象為單位,大大減少了渲染狀態(tài)切換。可是每次渲染的對(duì)象不同,因此每次都要將對(duì)象的頂點(diǎn)和索引數(shù)據(jù)拷貝到顯卡,會(huì)消耗不少時(shí)間。??
  可見想減少渲染狀態(tài)切換就要頻繁拷貝頂點(diǎn)索引數(shù)據(jù),而想減少拷貝頂點(diǎn)索引數(shù)據(jù)又不得不增加渲染狀態(tài)切換。魚與熊掌不可兼得 :-(??
  由于硬件條件和場(chǎng)景數(shù)據(jù)的情況比較復(fù)雜,具體哪種方法效率較高并沒有定式,兩種方法都有人使用,具體選用那種方法需要在實(shí)際環(huán)境測(cè)試后才能知道。??
? ?

 多光源問題??
待續(xù)……??

? ?

 陰影問題??
待續(xù)……??


 ??

渲染腳本??
  現(xiàn)在很多圖形程序都會(huì)自己定義一種腳本文件來描述Shader。??

  比如較早的OGRE(Object-oriented Graphics Rendering Engine,面向?qū)ο髨D形渲染引擎)的Material腳本,Quake3的Shader腳本,以及剛問世不久的Direct3D的Effect File,nVIDIA的CgFX腳本(文件格式與Direct3D Effect File兼容),ATI RenderMonkey使用的xml格式的腳本。OGRE Material和Quake3 Shader這兩種腳本比較有歷史了,不支持可編程渲染管線。而后面三種比較新的腳本都支持可編程渲染管線。??

? ?

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

? ?

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

可以非常方便的修改一個(gè)物體的外觀而不需重新編寫或編譯程序? ?

可以用外圍工具以所見即所得的方式來創(chuàng)建、修改腳本文件(類似ATI RenderMonkey的工作方式),便于美工、關(guān)卡設(shè)計(jì)人員設(shè)定對(duì)象外觀,建立外圍工具與圖形引擎的聯(lián)系? ?

可以在渲染時(shí)將相同外觀屬性及渲染狀態(tài)的對(duì)象(也就是Shader相同的對(duì)象)歸為一組,然后分組渲染,對(duì)每組對(duì)象只需要在渲染前設(shè)置一次渲染狀態(tài),大大減少了多余的狀態(tài)切換
posted on 2006-11-18 22:34 李錦俊(mybios) 閱讀(2030) 評(píng)論(0)  編輯 收藏 引用 所屬分類: Direct3D
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            国产精品天美传媒入口| 欧美 亚欧 日韩视频在线| 亚洲国产精品久久久久秋霞不卡 | 欧美日韩卡一卡二| 久久精品99国产精品| 正在播放亚洲| 亚洲精品一区二区三区婷婷月| 久久夜色精品亚洲噜噜国产mv| 亚洲在线观看视频| 亚洲六月丁香色婷婷综合久久| 黄色一区三区| 国产日韩欧美亚洲| 国产精品任我爽爆在线播放 | 国产亚洲精品aa午夜观看| 欧美精品一区二区三区久久久竹菊 | 久久婷婷蜜乳一本欲蜜臀| 亚洲性图久久| 99日韩精品| 日韩午夜高潮| 亚洲精品免费一二三区| 亚洲高清在线视频| 在线欧美日韩精品| 伊人久久婷婷| 在线看欧美日韩| 激情亚洲成人| 影音国产精品| 在线观看国产精品淫| 狠狠色狠狠色综合日日五| 国产欧美日韩视频一区二区| 国产精品婷婷| 国产精品视频一二| 国产精品一区二区三区乱码 | 亚洲午夜视频在线| 亚洲无线视频| 亚洲综合色自拍一区| 午夜一区二区三区不卡视频| 欧美亚洲在线| 久久天天综合| 欧美jizzhd精品欧美巨大免费| 美女主播一区| 欧美激情四色 | 久久精品女人天堂| 久久一二三四| 亚洲电影免费观看高清完整版在线 | 校园春色国产精品| 久久岛国电影| 久久视频精品在线| 欧美成人中文| 欧美吻胸吃奶大尺度电影| 国产精品久久久久aaaa樱花| 国产欧美一区二区精品秋霞影院| 国产午夜精品理论片a级大结局| 狠狠狠色丁香婷婷综合久久五月| 亚洲国产成人精品久久久国产成人一区| 亚洲成色777777在线观看影院| 亚洲激情欧美激情| 亚洲性夜色噜噜噜7777| 久久精品国产久精国产一老狼| 久久久综合精品| 亚洲大片在线| 亚洲一本大道在线| 久久一区精品| 欧美视频在线观看| 国内精品久久久久久影视8| 91久久久一线二线三线品牌| 亚洲伊人第一页| 久久夜色精品国产| 亚洲精品一区二区三| 亚洲欧美在线一区二区| 免费在线观看一区二区| 国产精品免费观看视频| 精品99视频| 亚洲一区视频在线观看视频| 久久青草福利网站| 99精品国产一区二区青青牛奶| 午夜精品久久久久| 欧美激情一区二区三区不卡| 国产日产欧产精品推荐色| 亚洲人成毛片在线播放| 性刺激综合网| 亚洲狠狠婷婷| 久久精品99无色码中文字幕| 欧美日韩一区二区在线 | 亚洲经典视频在线观看| 午夜精彩视频在线观看不卡| 欧美黑人多人双交| 国产综合久久久久影院| 亚洲一区日本| 最新国产乱人伦偷精品免费网站| 欧美亚洲视频| 欧美视频一区在线| 亚洲国产高清自拍| 久久国产直播| 中文在线资源观看网站视频免费不卡| 老色鬼精品视频在线观看播放| 国产精品免费一区豆花| 亚洲美女91| 欧美国产日韩在线观看| 羞羞色国产精品| 国产精品极品美女粉嫩高清在线| 亚洲精品一区二区三区婷婷月| 久久婷婷国产综合精品青草| 亚洲图色在线| 欧美三级电影一区| 亚洲免费成人av电影| 免费欧美在线视频| 久久精品国产2020观看福利| 国产精品视频yy9099| 亚洲无线一线二线三线区别av| 亚洲成色最大综合在线| 久久夜色精品亚洲噜噜国产mv| 国产一区激情| 久久精品国产久精国产爱| 亚洲性感美女99在线| 欧美三级第一页| 亚洲视频图片小说| 亚洲精品乱码视频| 欧美人妖另类| 99精品国产在热久久下载| 亚洲电影自拍| 欧美国产日韩在线| 亚洲美女黄色| 亚洲人成在线观看网站高清| 欧美国内亚洲| 一区二区不卡在线视频 午夜欧美不卡在 | 亚洲国产成人高清精品| 麻豆91精品| 亚洲国产精品福利| 欧美高清你懂得| 免费看精品久久片| 亚洲人成网站精品片在线观看| 欧美黄色网络| 欧美精品福利| 亚洲午夜日本在线观看| 中文成人激情娱乐网| 国产精品久久久一区麻豆最新章节| 亚洲在线不卡| 亚洲欧美在线另类| 国产专区欧美专区| 麻豆freexxxx性91精品| 老司机免费视频久久| 日韩午夜一区| 宅男66日本亚洲欧美视频| 国产精品女同互慰在线看| 欧美在线视频观看| 久久久久久黄| 最新国产の精品合集bt伙计| 亚洲黄网站在线观看| 欧美三级网址| 欧美一区二区三区在线看| 久久精品国产精品亚洲精品| 亚洲国产欧美一区| 日韩午夜黄色| 国产日韩欧美三级| 欧美jizzhd精品欧美巨大免费| 欧美激情第3页| 午夜精品久久久久久久99黑人| 欧美一区网站| 亚洲精品久久久久中文字幕欢迎你 | 国产精品制服诱惑| 久久综合久色欧美综合狠狠| 欧美/亚洲一区| 亚洲欧美日韩一区二区在线 | 亚洲自拍偷拍色片视频| 影音先锋中文字幕一区二区| 亚洲日本无吗高清不卡| 国产麻豆9l精品三级站| 欧美成人精品一区二区| 欧美三级视频在线播放| 久久久久看片| 欧美日韩精品免费观看| 久久久精品日韩| 欧美精品在线观看一区二区| 欧美一区二区三区四区高清| 美女国内精品自产拍在线播放| 亚洲午夜精品在线| 久久视频一区| 亚洲主播在线观看| 久久综合久久久久88| 午夜国产欧美理论在线播放| 美国成人直播| 欧美一区二区三区另类 | 亚洲午夜激情网页| 久久久www成人免费精品| 亚洲私人影院在线观看| 久久全球大尺度高清视频| 亚洲黄色影院| 欧美一区不卡| 亚洲一区久久| 欧美刺激午夜性久久久久久久| 久久精品国产99| 国产精品国产自产拍高清av| 欧美不卡在线| 国产在线一区二区三区四区 | 韩日欧美一区二区| 亚洲高清久久久| 狠色狠色综合久久| 亚洲欧美国产精品专区久久| 一本色道久久综合狠狠躁篇的优点| 久久精品国产99|