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

牽著老婆滿街逛

嚴以律己,寬以待人. 三思而后行.
GMail/GTalk: yanglinbo#google.com;
MSN/Email: tx7do#yahoo.com.cn;
QQ: 3 0 3 3 9 6 9 2 0 .

一種3D游戲碰撞檢測解決方案

轉載自:http://www.cnblogs.com/cproom/archive/2007/12/15/608732.html




        碰撞檢測在3D游戲中至關重要,好的碰撞檢測要求人物在場景中可以平滑移動,遇到一定高度內的臺階可以自動上去,而過高的臺階則把人擋住,遇到斜率較小的斜坡可以上去,斜率過大則把人擋住,在各種前進方向被擋住的情況下都要盡可能地讓人物沿合理的方向滑動而不是被迫停下。在滿足這些要求的同時還要做到足夠精確和穩定,防止人物在特殊情況下穿墻而掉出場景。

        碰撞檢測做得好了是應該的,不易被人注意到,因為這符合我們日常生活中的常識。做得差了卻很容易讓人發現,人物經常被卡住不能前進或者人物穿越了障礙。所以大部分人都覺得寫碰撞檢測代碼是件吃力不討好的事情,算法復雜、容易出bug、不容易出彩。下面還是回到正題,看看我們該如何解決這個難題。

        早期3D游戲的碰撞檢測多數基于格子或者BSP樹,基于格子的系統實現簡單但精度不夠,不屬于嚴格意義的3D碰撞檢測。基于BSP樹的碰撞檢測一度十分流行,算法基本已經成熟定型,但它的固有缺點卻使它不太適合現在的游戲。BSP樹需要很長的預處理時間不適合加載時計算,BSP劃分經常會產生原多邊形數三到四倍的多邊形,考慮到不用保存法線、顏色、uv等信息也要增加將近一倍的資源容量,在一個大的游戲中將模型資源的容量從200M增加到400M相信是大部分人都不愿接受的。目前對于任意復雜三角形集合(mesh)的碰撞檢測多數基于BVTreebounding volume tree,具體可以是aabb treeobb tree或者K-dop tree,這也是當今各種物理引擎和碰撞檢測引擎流行的做法。

        上面是碰撞檢測按數據結構不同的分類,按檢測方式又可以分為離散點的碰撞檢測和連續碰撞檢測(CCD continuous collision detection)。離散點的碰撞檢測是指定某一時刻T的兩個靜態碰撞體,看它們之間是否交迭,如果沒有交迭則返回它們最近點的距離,如果交迭則返回交迭深度,交迭方向等。連續碰撞檢測則是分別指定在T1T2兩個時刻兩個碰撞體的位置,看它們在由T1運動到T2時刻的過程中是否發生碰撞,如果碰撞則返回第一碰撞點的位置和法線。連續碰撞檢測是最為自然的碰撞檢測,可以大大方便碰撞響應邏輯的編寫,可以很容易避免物體發生交迭或者穿越。離散點的碰撞檢測則沒有那么友好,當檢測到碰撞時兩個物體已經發生了交迭,如果其中有三角形網格對象那么已經有許多三角形發生了交迭,如何將兩個交迭的對象分開并按合理的方式運動是一個挑戰。雖然連續碰撞檢測是最自然的方式,但它的實現非常復雜,運算開銷也很大,所以目前大部分成熟的物理引擎和碰撞檢測引擎還是采用了基于離散點的碰撞檢測,為了避免物體交迭過深或者彼此穿越,它們都要采用比較小的模擬步長。

        由于碰撞檢測引擎的復雜性和對效率的高要求,我們應該盡量使用目前成熟的完整引擎,而不是自己去開發。經過評估,我決定采用Opcode碰撞檢測引擎來做游戲中人物和場景的碰撞檢測。Opcode的主要功能是用aabb tree管理復雜三角形集合來和射線、球體,立方體,另一個三角形集合等進行離散點上的碰撞檢測,如果檢測到交迭則返回所有發生交迭的三角形。Opcode的特點是高度的內存使用優化和極好的效率,ODE物理引擎底層就采用它來做復雜三角形mesh的碰撞檢測,Opcode的作者也是NovodeXPhysX前身)物理引擎的核心開發人員,據說NovodeX采用了Opcode的一個更優化版本。由此可見Opcode的成熟與效率。

        確定了要使用的引擎,下面要討論的算法就和具體引擎無關了,適合于任何離散點的碰撞檢測引擎。我們用AABB包圍盒來代表場景中的人物,看看如何實現文章開頭所提出的效果。

         首先解釋一下檢測地面的方式,沿人物包圍盒的四條豎邊向下投四條射線,射線的終點略低于人物的腳底(也就是說射線的長度是有限的),如果與場景發生碰撞并且碰撞平面的斜率小于某一值則取得碰撞點的高度,否則視為這條射線沒有檢測到地面。將所有射線檢測到的地面高度最大值作為最后的地面高度,如果四條射線都沒有檢測到地面則認為人物懸空。

 

vD = 當前幀人物位移

p0 
= 人物包圍盒中心當前位置

bOnGroundP1; 
// 人物是否站在地面

bOnGroundP3; 
// 人物是否站在地面

bOnGround; 
// 人物是否站在地面

 

p1 
= p0 + vD

在p1位置檢測地面

if( 檢測到地面 )

{

     將包圍盒下放到地面得到位置p2

     bOnGroundP1 
= true;

}


else

{

     p2 
= p1;

     bOnGroundP1 
= false;

}





測試p2點的包圍盒是否與場景交迭

if( 交迭 )

{

     取得所有交迭三角形的法線,將它們相加后取平均值,得到法線normal

     將法線與向上的向量叉乘得到切線方向tangent

     
// 計算人物沿切線滑動后的位置,注意這里用p0做計算。

     
// 如果要使滑動更平滑可以把p0向法線方向推出一些

     
// p3 = p0 + normal * 0.1f + vD.Dot(tangent);

     p3 
= p0 + vD.Dot(tangent); 

 

     在p3位置檢測地面

     
if( 檢測到地面 )

     
{

         將包圍盒下放到地面得到位置p4

          bOnGroundP3 
= true;

     }


     Else

     
{

         p4 
= p3;

          bOnGroundP3 
= false;

     }


 

     測試p4點的包圍盒是否與場景交迭

     
if( 交迭 )

     
{

         測試p1點的包圍盒是否與場景交迭

         
if( 交迭 )

         
{

              
// 無法得到合理的移動位置,人物位置保持不變

              
// 等待下一幀玩家調整前進方向再做測試

              將p0作為人物的新位置

              bOnGround 
= true;

              
return;

         }


         
else

         
{

              將p1作為人物的新位置

              bOnGround 
= bOnGroundP1;

              
return;

         }


     }


     Else

     
{

         將p4作為人物的新位置

         bOnGround 
= bOnGroundP3;

          
return;

     }


}


else

{

     將p2作為人物的新位置

     bOnGround 
= bOnGroundP1;

     
return;

}

 

 

上面的算法基本達到了文章開頭所提到的效果,在某些復雜情況下人物移動還有些不流暢,但沒有發現人物有穿越障礙物的現象。在大部分情況下人物的新坐標都會由p2點返回,最壞情況是人物被卡住返回p0點。在P4 3.0G的機器上可以支持120個人物在最壞情況下保持30幀的游戲幀數。

posted on 2007-12-29 12:08 楊粼波 閱讀(926) 評論(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>
            美女主播视频一区| 女人天堂亚洲aⅴ在线观看| 亚洲经典自拍| 亚洲第一福利社区| 亚洲欧洲一区二区三区| 亚洲激情女人| 亚洲视频在线观看网站| 午夜精品久久久久影视 | 亚洲人成在线播放网站岛国| 亚洲国产日韩在线| 国产精品99久久不卡二区| 性欧美videos另类喷潮| 嫩草影视亚洲| 亚洲作爱视频| 欧美资源在线观看| 欧美14一18处毛片| 国产精品久久福利| 伊人久久综合| 亚洲综合99| 农村妇女精品| 亚洲桃色在线一区| 久久久亚洲精品一区二区三区| 欧美高清在线播放| 国产亚洲一区精品| 在线中文字幕不卡| 六月婷婷久久| 一区二区久久久久| 久久亚洲国产精品一区二区| 欧美日韩视频第一区| 激情一区二区三区| 亚洲欧美日韩中文播放| 欧美激情一区二区三区四区 | 噜噜噜噜噜久久久久久91 | 国产精品成人观看视频免费| 国产一区在线看| 中文在线资源观看网站视频免费不卡 | 亚洲欧美福利一区二区| 两个人的视频www国产精品| 99国产麻豆精品| 欧美在线亚洲一区| 国产精品欧美日韩| 亚洲天堂免费观看| 亚洲电影免费在线观看| 亚洲欧美日韩成人| 欧美日韩另类字幕中文| 亚洲国产精品久久久久秋霞不卡 | 一色屋精品视频免费看| 欧美成人午夜免费视在线看片| 久久国产66| 国产精品久久久久久久久久免费看 | 美日韩精品视频| 国产一区二区三区高清| 亚洲无亚洲人成网站77777| 欧美国产视频一区二区| 久久精品国产一区二区三区| 国产日产欧产精品推荐色| 亚洲一区免费看| avtt综合网| 欧美日韩午夜剧场| 亚洲剧情一区二区| 亚洲精品久久久久久下一站| 欧美成人情趣视频| 亚洲精品日日夜夜| 亚洲国产欧美在线人成| 欧美国产日韩一二三区| 一区二区免费在线视频| 一本久久精品一区二区| 国产精品美女在线| 久久久精品动漫| 久久久久久久综合日本| 亚洲片在线观看| 日韩午夜在线| 国产精品永久免费| 久久阴道视频| 欧美成年人网| 亚洲私拍自拍| 午夜精品av| 亚洲国产精品黑人久久久| 亚洲高清资源| 国产精品国产三级国产专播品爱网| 亚洲一区中文| 久久成人综合网| 亚洲精品久久久久久下一站 | 亚洲午夜在线观看视频在线| 亚洲免费视频一区二区| 黄色亚洲精品| 日韩视频免费看| 国产一区二区成人久久免费影院| 久久永久免费| 欧美日韩国产综合久久| 欧美中文字幕久久| 美女主播精品视频一二三四| 亚洲桃花岛网站| 久久爱另类一区二区小说| 亚洲日韩欧美视频| 亚洲自拍电影| 亚洲精品免费一区二区三区| 亚洲一区二区影院| 亚洲经典视频在线观看| 亚洲愉拍自拍另类高清精品| 亚洲国产欧美一区二区三区丁香婷| 一本色道久久综合亚洲二区三区| 久久久久在线| 国产精品区二区三区日本 | 亚洲美女在线观看| 亚洲一区二区三区乱码aⅴ蜜桃女| 精品动漫3d一区二区三区免费| 亚洲美女网站| 亚洲第一在线视频| 午夜欧美大尺度福利影院在线看| 亚洲国产一区在线观看| 亚洲欧美日韩国产| 在线一区视频| 欧美激情成人在线视频| 老司机67194精品线观看| 国产精品爱久久久久久久| 欧美激情导航| 在线欧美视频| 久久久久99| 久久视频精品在线| 国产婷婷一区二区| 亚洲一区二区三区免费视频| 亚洲午夜在线观看| 欧美日韩一区二区三区在线看| 欧美激情亚洲自拍| 亚洲国产精品123| 久久国产精品免费一区| 欧美专区在线播放| 国产精品九色蝌蚪自拍| 中文国产成人精品| 亚洲先锋成人| 国产精品盗摄久久久| 夜夜爽99久久国产综合精品女不卡| 亚洲精品一级| 免费欧美视频| 欧美~级网站不卡| 亚洲国产美女| 欧美va亚洲va香蕉在线| 欧美国产精品劲爆| 亚洲激情第一页| 欧美极品aⅴ影院| 亚洲精品乱码久久久久久久久| 亚洲激情国产精品| 欧美久久成人| 一卡二卡3卡四卡高清精品视频 | 欧美福利精品| 亚洲精品韩国| 欧美日韩在线视频首页| 亚洲午夜电影| 久久综合九色综合网站| 精品99视频| 欧美福利小视频| 一区二区三区高清不卡| 午夜精品一区二区三区在线视| 国产精品每日更新| 欧美一区免费视频| 免费中文日韩| 亚洲一区二区三区视频播放| 国产麻豆91精品| 久久综合福利| 99精品视频免费| 欧美一区二区三区视频免费| 一区二区亚洲欧洲国产日韩| 欧美激情视频在线播放 | 欧美一区二区三区男人的天堂| 久久综合影视| 久久av一区二区三区| 亚洲色图自拍| 国产亚洲一区二区三区| 免费看黄裸体一级大秀欧美| 日韩一级精品视频在线观看| 午夜亚洲伦理| 亚洲欧洲一区二区天堂久久| 国产精品高潮久久| 毛片av中文字幕一区二区| 一区二区久久| 亚洲国产日韩一区| 久久精品99国产精品| 一本色道久久88亚洲综合88| 韩国av一区二区三区| 欧美午夜精品久久久久久超碰| 久久天堂成人| 亚欧成人精品| 亚洲天堂av在线免费| 欧美高清一区| 久久久久久夜| 午夜精品999| 夜夜嗨av色一区二区不卡| 韩日精品在线| 国产嫩草一区二区三区在线观看| 欧美高清视频| 欧美ab在线视频| 久久亚洲精品中文字幕冲田杏梨| 一区二区三区高清| 最新国产成人av网站网址麻豆| 另类成人小视频在线| 久久精品91| 欧美在线观看视频一区二区| 亚洲在线一区| 亚洲免费视频观看|