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

游戲開發中常用的設計模式

Posted on 2008-07-09 15:34 RichardHe 閱讀(448) 評論(0)  編輯 收藏 引用 所屬分類: OGRE

出自http://blog.csdn.net/duzhi5368/archive/2008/04/22/2314232.aspx

使用設計模式來提高程序庫的重復利用性是大型程序項目開發必須的。但是在“四人幫”的設計模式概述中提到了23種標準設計模式,不但難以記住,而且有些設計模式更多的適用于應用程序開發,對游戲項目引擎設計并沒有很多的利用價值。根據經驗,精挑細選后,篤志在這里記錄一些自認為有利用價值的設計模式,以便之后自己設計時使用。

 

一:觀察者Observer

 

觀察者的設計意圖和作用是 它將對象與對象之間創建一種依賴關系,當其中一個對象發生變化時,它會將這個變化通知給與其創建關系的對象中,實現自動化的通知更新。

 

       游戲中觀察者的適用環境有

1UI控件管理類。當我們的GUI控件都使用觀察者模式后,那么用戶的任何界面相關操作和改變都將會通知其關聯對象-----我們的UI事件機。

2:動畫管理器。很多時候我們在播放一個動畫楨的時候,對其Frame有很大興趣,此時我們設置一個FrameLister對象對其進行監視,獲得我們關心的事件進行處理是必須的。

 

觀察者偽代碼

//-------------------------------------------------------------------------------------------------------

// 被觀察對象目標類

Class Subject

{

       // 對本目標綁定一個觀察者 Attach( Observer );

       // 解除一個觀察者的綁定   DeleteAttach( Observer );

       // 本目標發生改變了,通知所有的觀察者,但沒有傳遞改動了什么

       Notity()

       {

              For ( …遍歷整個ObserverList …)

              { pObserver ->Update(); }

}

// 對觀察者暴露的接口,讓觀察者可獲得本類有什么變動GetState();

}

//-------------------------------------------------------------------------------------------------------

// 觀察者/監聽者類

Class Observer

{

       // 暴露給對象目標類的函數,當監聽的對象發生了變動,則它會調用本函數通知觀察者

Void Update ()

{

pSubject ->GetState();  // 獲取監聽對象發生了什么變化

TODODisposeFun();  // 根據狀態不同,給予不同的處理

}

}

//-------------------------------------------------------------------------------------------------------

 

非程序語言描述

AB的好朋友,對B的行為非常關心。B要出門,此時A給了B一個警報器,告訴B說:“如果你有事,立刻按這個警報器告訴我。”。結果B在外面遇上了麻煩,按下警報器(Update()),B就知道A出了事,于是就調查一下B到底遇到了什么麻煩(GetState()),當知道B原來是因為被人打了,于是立刻進行處理DisposeFun(),派了一群手下幫B打架。

當然關心A的人可以不止一個,CD可能也對A很關心,于是A這里保存一個所有關心它的人的鏈表,當遇到麻煩的時候,輪流給每個人一份通知。

 

二:單件模式Singleton

單件模式的設計意圖和作用是    保證一個類僅有一個實例,并且,僅提供一個訪問它的全局訪問點。

 

游戲中適用于單件模式的有

1:所有的Manger。在大部分的流行引擎中都存在著它的影子,例如SoundManager, ParticeManager等。

2:大部分的工廠基類。這一點在大部分引擎中還是見不到的,實際上,我們的父類工廠采用唯一實例的話,我們子類進行擴展時也會有很大方便。

 

單件模式偽代碼

//-------------------------------------------------------------------------------------------------------

Class Singleton

{

       Static MySingleton;       // 單件對象,全局唯一的。

       Static Instance(){ return MySingleton;}              // 對外暴露接口

}

//-------------------------------------------------------------------------------------------------------

 

三:迭代器Iterator

      

       迭代器設計意圖和作用是    提供一個方法,對一個組合聚合對象內各個元素進行訪問,同時又不暴露該對象類的內部表示。

 

       游戲中適用于迭代器模式的有    因為STL的流行,這個設計已經廣為人知了,我們對任何形式的資源通一管理時,不免會將其聚合起來,或者List,或者Vector,我們都需要一個對其進行訪問的工具,迭代器無疑是一個利器。

 

       迭代器偽代碼

//-------------------------------------------------------------------------------------------------------

// 迭代器基類

       Class Iterator

{

              Virtual First();              

              Virtual Next();

              Virtual End();

              Virtual CurrentItem();    // 返回當前Item信息

}

//-------------------------------------------------------------------------------------------------------

// 聚合體的基類

       Class ItemAggregate

{

              Virtual CreateIterator();  // 創建訪問自身的一個迭代器

}

//-------------------------------------------------------------------------------------------------------

// 實例化的項目聚合體

       Class InstanceItemAggregate : public ItemAggregate

       {

              CreateIterator(){ return new InstanceIterator(this); }

}

//-------------------------------------------------------------------------------------------------------

 

四:訪問者模式Visitor

 

       訪問者設計意圖和作用是    當我們希望對一個結構對象添加一個功能時,我們能夠在不影響結構的前提下,定義一個新的對其元素的操作。(實際上,我們只是把對該元素的操作分割給每個元素自身類中實現了而已)

 

       游戲中適用于訪問者模式的有    任何一個比較靜態的復雜結構類中都適合采用一份訪問者。這里的“比較靜態的復雜結構類”意思是,該結構類中元素繁多且種類復雜,且對應的操作較多,但類很少進行變化,我們就能夠將,對這個結構類元素的操作獨立出來,避免污染這些元素對象。

       1:例如場景管理器中管理的場景節點,是非常繁多的,而且種類不一,例如有Ogre中的Root, Irrchit中就把攝象機,燈光,Mesh,公告版,聲音都做為一種場景節點,每個節點類型是不同的,雖然大家都有共通的Paint(),Hide()等方法,但方法的實現形式是不同的,當我們外界調用時需要統一接口,那么我們很可能需要需要這樣的代碼

       Hide( Object )

       { if (Object == Mesh) HideMesh();  if (Object == Light) HideLight();    }

此時若我們需要增加一個Object新的類型對象,我們就不得不對該函數進行修正。而我們可以這樣做,讓Mesh,Light他們都繼承于Object,他們都實現一個函數Hide(),那么就變成

       Mesh::Hide( Visitor ) { Visitor.Hide (Mesh); }

       Light::Hide(Visitor ){ Visitor.Hide (Light); }

我們在調用時只需要Object.Hide(Visitor){ return Visitor.Hide(Object); }

 

這樣做的好處,我們免去了對重要函數的修正,Object.Hide(Visitor){}函數我們可以永久不變,但是壞處也是很明顯的,因為將方法從對象集合結構中抽離出來,就意味著我們每增加一個元素,它必須繼承于一個抽象的被訪問者類,實現其全部函數,這個工作量很大。

所以,訪問者是僅適合于一個裝載不同對象的大容器,但同時又要求這個容器的元素節點不應當有大的變動時才使用。另外,廢話一句,訪問者破壞了OO思想的。

 

       訪問者偽代碼

//-------------------------------------------------------------------------------------------------------

//  訪問者基類

       Class Visitor

{

              Virtual VisitElement( A ){ … };             // 訪問的每個對象都要寫這樣一個方法

              Virtual VisitElement( B ){ … };

}

 

// 訪問者實例A

Class VisitorA

{

       VisitElement( A ){ … };         // 實際的處理函數

VisitElement( B ){ … };        // 實際的處理函數

}

 

// 訪問者實例B

Class VisitorB

{

       VisitElement( A ){ … };         // 實際的處理函數

VisitElement( B ){ … };        // 實際的處理函數

}

 

 

// 被訪問者基類

Class Element

{

       Virtual Accept( Visitor );        // 接受訪問者

}

 

// 被訪問者實例A

Class ElementA

{

       Accecpt( Visitor v ){ v-> VisitElement(this); };    // 調用注冊到訪問者中的處理函數

}

 

// 被訪問者實例B

Class ElementB

{

       Accecpt( Visitor v ){ v-> VisitElement(this); };    // 調用注冊到訪問者中的處理函數

}

 

//-------------------------------------------------------------------------------------------------------

 

五:外觀模式Façade

      

       外觀模式的設計意圖和作用是   將用戶接觸的表層和內部子集的實現分離開發。實際上,這個模式是個紙老虎,之后我們看偽代碼立刻就會發現,這個模式實在用的太頻繁了。

 

       游戲中需要使用外觀模式的地方是   這個非常多了,舉幾個比較重要的。

       1:實現平臺無關性。跨平臺跨庫的函數調用。

       2:同一個接口去讀取不同的資源。

       3:硬件自動識別處理系統。

 

       外觀模式偽代碼

//-------------------------------------------------------------------------------------------------------

       // 用戶使用的接口類

       Class Interface

{

// 暴露出來的函數接口函數,有且僅有一個,但內部實現是調用了兩個類

       Void InterfaceFun()

{

       // 根據某種條件,底層自主的選擇使用AB的方法。用戶無須關心底層實現

       If ( XXX )

       {

              ActualA->Fun();

}

Else

{

       ActualB->Fun();

}

};   

}

 

// 實際的實現,不暴露給用戶知道

Class ActualA

{

       Void Fun();

}

 

// 實際的實現,不暴露給用戶知道

Class ActualB

{

       Void Fun();

}

 

怎么樣,紙老虎吧,看起來很高深摸測的命名而已。

//-------------------------------------------------------------------------------------------------------

 

六:抽象工廠模式AbstractFactory

      

       抽象工廠的設計意圖和作用是    封裝出一個接口,這個接口負責創建一系列互相關聯的對象,但用戶在使用接口時不需要指定對象所在的具體的類。從中文命名也很容易明白它是進行批量生產的一個生產工廠的作用。

 

       游戲中使用抽象工廠的地方有    基本上任何有批量的同類形式的子件地方就會有工廠的存在。(補充一句:下面代碼中的ConcreteFactory1實例工廠就是工廠,而抽象工廠僅僅是工廠的一個抽象層而已。

1:例如,在音頻方面,一個音頻的抽象工廠派生出不同的工廠,有音樂工廠,音效工廠。音效工廠中又有一個創建3D音效節點的方法,一個創建普通音效節點的方法。最終用戶只需要SoundFactory->Create3DNode( pFileName );就可以創建一個節點了。

2:場景對象。

3:渲染對象。

4:等等……

工廠與單件,管理器Manager關系一定是非常緊密的。

 

       抽象工廠偽代碼

//-------------------------------------------------------------------------------------------------------

       class AbstractProductA {}; // 抽象的產品A基類
  class AbstractProductB {}; //抽象的產品B基類

 

// 抽象工廠基類
  class AbstractFactory
  
{
  
public:
   virtual AbstractProductA* CreateProductA() = 0 ;// 創建
ProductA
   virtual AbstractProductB* CreateProductB() = 0 ;// 創建
ProductB
       } ;

 

  class ProductA1 : public AbstractProductA {};    // 產品A的實例1
  class ProductA2 : public AbstractProductA {};    // 產品A的實例2

  class ProductB1 : public AbstractProductB {};    // 產品B的實例1
  class ProductB2 : public AbstractProductB {};    // 產品B的實例2

 

// 實例工廠1

      class ConcreteFactory1 : public AbstractFactory
  
{
    
virtual AbstractProductA* CreateProductA() { return new ProductA1() ; }
    
virtual AbstractProductB* CreateProductB() { return new ProductB1() ; }
       static ConcreteFactory1* Instance() { }        // 實例工廠盡量使用單件模式

  } ;

 

// 實例工廠2

class ConcreteFactory2 : public AbstractFactory
  {
    
virtual AbstractProductA* CreateProductA() { return new ProductA2() ; }
    
virtual AbstractProductB* CreateProductB() { return new ProductB2() ; }
    static ConcreteFactory2* Instance() {} 
       // 實例工廠盡量使用單件模式
  } ;

}

//-------------------------------------------------------------------------------------------------------

客戶端代碼

Void main()
{
  AbstractFactory *pFactory1 = ConcreteFactory1::Instance() ;
  
AbstractProductA *pProductA1 = pFactory1->CreateProductA() ;
  
AbstractProductB *pProductB1 = pFactory1->CreateProductB() ;
  
AbstractFactory *pFactory2 = ConcreteFactory2::Instance() ;
  
AbstractProductA *pProductA2 = pFactory2->CreateProductA() ;
  
AbstractProductB *pProductB2 = pFactory2->CreateProductB() ;
}

//-------------------------------------------------------------------------------------------------------

posts - 94, comments - 138, trackbacks - 0, articles - 94

Copyright © RichardHe

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美精品18| 国产精品久久久久久久久借妻| 国产日韩精品入口| 久久在线播放| 久久久久国产精品厨房| 亚洲一区成人| 亚洲欧美日韩一区| 亚洲综合日本| 久久久五月天| 久久综合99re88久久爱| 久久精品人人做人人综合| 欧美一区观看| 美日韩在线观看| 亚洲精品一区中文| 亚洲欧美日韩视频一区| 久久影院午夜论| 午夜亚洲福利在线老司机| 欧美高清视频在线| 久久久欧美精品| 一区二区三区国产精品| 久久精品国产第一区二区三区最新章节| 蜜臀久久久99精品久久久久久| 欧美三级电影大全| 亚洲欧洲久久| 久久久久久自在自线| 亚洲美女福利视频网站| 美女视频一区免费观看| 久久精品国产欧美激情| 亚洲一二三区在线观看| 免费看精品久久片| 亚洲精品久久嫩草网站秘色| 久久av二区| 欧美制服第一页| 亚洲一区二区三区三| 日韩一级精品| 亚洲片在线资源| 久久蜜桃资源一区二区老牛 | 久久久午夜电影| 夜夜爽av福利精品导航| 亚洲电影免费观看高清完整版在线| 欧美一区二区三区精品| 国产日韩欧美精品综合| 亚洲影视在线播放| 午夜精品久久久久久99热| 国产色视频一区| 欧美成人午夜剧场免费观看| 久久深夜福利免费观看| 99精品国产福利在线观看免费| 欧美日本久久| 欧美精品日韩综合在线| 亚洲五月婷婷| 久久精品国产第一区二区三区最新章节 | 在线免费观看一区二区三区| 久久亚洲春色中文字幕久久久| 久久综合网色—综合色88| 在线观看不卡| 99精品福利视频| 今天的高清视频免费播放成人| 欧美成人官网二区| 亚洲欧美日韩在线不卡| 久久精品在线观看| 欧美大片专区| 久久综合影视| 国产精品久久久爽爽爽麻豆色哟哟| 欧美一区二区视频在线观看| 欧美freesex8一10精品| 亚洲免费影视第一页| 看欧美日韩国产| 国产欧美不卡| 国产精品99久久久久久久vr | 欧美剧在线免费观看网站| 欧美一区二区三区精品 | 亚洲国产精品久久久| 久久久久免费| 亚洲视屏一区| 欧美精品久久久久久久久老牛影院 | 欧美一区1区三区3区公司| 一区二区三区国产在线| 欧美成人r级一区二区三区| 久久夜色撩人精品| 91久久精品国产91性色tv| 久久亚洲一区二区三区四区| 狼狼综合久久久久综合网| 国内精品久久久久影院色 | 国产精品久久久久久久浪潮网站| 欧美国产日韩在线观看| 久久一日本道色综合久久| 性久久久久久| 欧美高清在线一区二区| 亚洲日本va在线观看| 欧美高清在线视频| 一区二区91| 欧美黄色aaaa| 午夜精品久久久久久久99樱桃| 国产精品国产三级国产aⅴ浪潮 | 一区二区高清视频| 国产精品av久久久久久麻豆网 | 亚洲一级二级在线| 国产精品一卡二卡| 欧美激情偷拍| 欧美专区一区二区三区| 久久视频在线看| 国产欧美va欧美va香蕉在| 美女主播一区| 亚洲欧美日韩国产综合精品二区 | 国产麻豆综合| 欧美日韩久久久久久| 99视频精品全部免费在线| 国产原创一区二区| 欧美性jizz18性欧美| 欧美jizz19性欧美| 久久美女艺术照精彩视频福利播放| 一本色道久久99精品综合 | 91久久精品一区| 国产精品自在欧美一区| 国产精品毛片a∨一区二区三区|国| 夜夜嗨av一区二区三区| 91久久精品视频| 欧美三级免费| 欧美日韩综合久久| 欧美色大人视频| 国产精品国产三级国产普通话99| 欧美日韩亚洲一区二区三区| 欧美日韩一区二区三区在线看 | 国产精品久久久久秋霞鲁丝| 久久久久久综合网天天| 亚欧成人在线| 欧美激情视频网站| 一本色道久久综合亚洲精品按摩| 亚洲精品中文字幕在线| 亚洲综合日韩| 欧美美女bb生活片| 欧美午夜免费电影| 久久久久久午夜| 在线观看欧美成人| 夜夜嗨av一区二区三区网页| 亚洲一区二区成人| 欧美一区国产一区| 亚洲第一狼人社区| 欧美成人免费网站| 亚洲综合国产精品| 欧美日韩国产一级| 亚洲电影有码| 久久精品成人一区二区三区 | 揄拍成人国产精品视频| 亚洲午夜一区二区| 欧美好骚综合网| 久久琪琪电影院| 国内精品久久久久影院色 | 99re66热这里只有精品3直播| 欧美激情第六页| 久久夜色撩人精品| 欧美伊人久久大香线蕉综合69| 欧美性猛交xxxx免费看久久久 | 亚洲福利免费| 久久久久久久性| 久久精品免费观看| 激情六月婷婷综合| 久久综合国产精品台湾中文娱乐网| 亚洲一区二区三区高清不卡| 日韩系列在线| 欧美视频在线观看一区二区| 欧美一区二区三区久久精品茉莉花| 亚洲夜间福利| 影音先锋久久资源网| 亚洲人成在线观看一区二区| 欧美日韩三区| 久久伊人精品天天| 欧美精品综合| 久久国产精品72免费观看| 欧美淫片网站| 一区二区三区国产精华| 欧美伊人影院| 亚洲在线一区| 免费观看成人www动漫视频| 99视频一区二区| 日韩视频在线观看国产| 欧美日韩一区二区三区四区在线观看 | 亚洲欧美一区二区原创| 黄色日韩在线| 国产精品视频导航| 国产精品海角社区在线观看| 亚洲欧美另类国产| 久久精品国产精品亚洲| 亚洲视频一二| 久久中文久久字幕| 亚洲欧美另类在线| 麻豆精品91| 久久爱www久久做| 亚洲区欧美区| 午夜在线一区| 亚洲桃色在线一区| 西西人体一区二区| 99精品福利视频| 欧美激情精品| 亚洲国产精品成人综合色在线婷婷| 夜夜爽99久久国产综合精品女不卡| 国产噜噜噜噜噜久久久久久久久| 欧美福利小视频| 亚洲激情视频|