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

[轉(zhuǎn)載]低耦合模塊間的通信組件:兩個(gè)模板

用途

在一個(gè)UI與邏輯模塊交互比較多的程序中,因?yàn)椴⒉幌胱寖蓚€(gè)模塊發(fā)生太大的耦合,基本目標(biāo)是
可以完全不改代碼地?fù)Q一個(gè)UI。邏輯模塊需要在產(chǎn)生一些事件后通知到UI模塊,并且在這個(gè)通知
里攜帶足夠多的信息(數(shù)據(jù))給接收通知的模塊,例如UI模塊。邏輯模塊還可能被放置于與UI模
塊不同的線程里。

最初的結(jié)構(gòu)

最開始我直接采用最簡(jiǎn)單的方法,邏輯模塊保存一個(gè)UI模塊傳過來的listener。當(dāng)有事件發(fā)生時(shí),
就回調(diào)相應(yīng)的接口將此通知傳出去。大致結(jié)構(gòu)如下:

 /// Logic
 class EventNotify
 
{
 
public:
  
virtual void OnEnterRgn( Player *player, long rgn_id );
 }
;

 
/// UI
 class EventNotifyImpl : public EventNotify
 
{
 }
;

 
/// Logic
 GetEventNotify()->OnEnterRgn( player, rgn_id );

 

但是,在代碼越寫越多之后,邏輯模塊需要通知的事件越來越多之后,EventNotify這個(gè)類開始
膨脹:接口變多了、不同接口定義的參數(shù)看起來也越來越惡心了。

改進(jìn)

于是我決定將各種事件通知統(tǒng)一化:

 

struct Event
{
 
long type; // 事件類型
  // 附屬參數(shù)
}
;

 

這樣,邏輯模塊只需要?jiǎng)?chuàng)建事件結(jié)構(gòu),兩個(gè)模塊間的通信就只需要一個(gè)接口即可:

void OnNotify( const Event &event );

但是問題又來了,不同的事件類型攜帶的附屬參數(shù)(數(shù)據(jù))不一樣。也許,可以使用一個(gè)序列化
的組件,將各種數(shù)據(jù)先序列化,然后在事件處理模塊對(duì)應(yīng)地取數(shù)據(jù)出來。這樣做總感覺有點(diǎn)大動(dòng)
干戈了。當(dāng)然,也可以使用C語言里的不定參數(shù)去解決,如:

void OnNotify( long event_type, ... )

其實(shí),我需要的就是一個(gè)可以表面上類型一樣,但其內(nèi)部保存的數(shù)據(jù)卻多樣的東西。這樣一想,
模塊就能讓事情簡(jiǎn)單化:

 

template <typename P1, typename P2>
class Param
{
public:
 Param( P1 p1, P2 p2 ) : _p1( p1 ), _p2( p2 )
 
{
 }

 
 P1 _p1;
 P2 _p2;
}
;

template 
<typename P1, typename P2>
void OnNotify( long event_type, Param<P1, P2> param );

GetNotify()
->OnNotify( ET_ENTER_RGN, Param<Player*long>( player, rgn_id ) );
GetNotify()
->OnNotify( ET_MOVE, Param<longlong>( x, y ) );

 

在上面這個(gè)例子中,雖然通過Param的包裝,邏輯模塊可以在事件通知里放置任意類型的數(shù)據(jù),但
畢竟只支持2個(gè)參數(shù)。實(shí)際上為了實(shí)現(xiàn)支持多個(gè)參數(shù)(起碼得有15個(gè)),還是免不了自己實(shí)現(xiàn)多個(gè)
參數(shù)的Param。

幸虧我以前寫過宏遞歸產(chǎn)生代碼的東西,可以自動(dòng)地生成這種情況下諸如Param1、Param2的代碼。
如:

 

#define CREATE_PARAM( n ) \
 template 
<DEF_PARAM( n )> \
 
struct Param##n \
 
{ \
  DEF_PARAM_TYPE( n ); \
  Param##n( DEF_FUNC_PARAM( n ) ) \
  
{ \
   DEF_MEM_VAR_ASSIGN( n ); \
  }
 \
  DEF_VAR_DEF( n ); \
 }


 CREATE_PARAM( 
1 );
 CREATE_PARAM( 
2 );

 

即可生成Param1和Param2的版本。其實(shí)這樣定義了Param1、Param2的東西之后,又使得OnNotify
的參數(shù)不是特定的了。雖然可以把Param也泛化,但是在邏輯層寫過多的模板代碼,總感覺不好。

于是又想到以前寫的一個(gè)東西,可以把各種類型包裝成一種類型---對(duì)于外界而言:any。any在
boost中有提到,我只是實(shí)現(xiàn)了個(gè)簡(jiǎn)單的版本。any的大致實(shí)現(xiàn)手法就是在內(nèi)部通過多態(tài)機(jī)制將各
種類型在某種程度上隱藏,如:

 

        class base_type
        
{
        
public:
            
virtual ~base_type()
            
{
            }

            
virtual base_type *clone() const = 0;
        }
;
        
        template 
<typename _Tp>
        
class var_holder : public base_type
        
{
        
public:
            typedef _Tp type;
            typedef var_holder
<type> self;
        
public:
            var_holder( 
const type &t ) : _t( t )
            
{
            }


            base_type 
*clone() const
            
{
                
return new self( _t );
            }

        
public:
            type _t;
        }


這樣,any類通過一個(gè)base_type類,利用C++多態(tài)機(jī)制即可將類型隱藏于var_holder里。那么,
最終的事件通知接口成為下面的樣子:

void OnNotify( long type, any data );

OnNotify( ET_ENTER_RGN, any( create_param( player, rgn_id ) ) );其中,create_param
是一個(gè)輔助函數(shù),用于創(chuàng)建各種Param對(duì)象。

事實(shí)上,實(shí)現(xiàn)各種ParamN版本,讓其名字不一樣其實(shí)有點(diǎn)不妥。還有一種方法可以讓Param的名字
只有一個(gè),那就是模板偏特化。例如:

 

template <typename _Tp>
struct Param;

template 
<>
struct Param<void()>;

template 
<typename P1>
struct Param<void(P1)>

template 
<typename P1, typename P2>
struct Param<void(P1,P2)>

 

這種方法主要是通過組合出一種函數(shù)類型,來實(shí)現(xiàn)偏特化。因?yàn)槲矣X得構(gòu)造一個(gè)函數(shù)類型給主模版,
并不是一種合情理的事情。但是,即使使用偏特化來讓Param名字看起來只有一個(gè),但對(duì)于不同的
實(shí)例化版本,還是不同的類型,所以還是需要any來包裝。

實(shí)際使用

實(shí)際使用起來讓我覺得非常賞心悅目。上面做的這些事情,實(shí)際上是做了一個(gè)不同模塊間零耦合
通信的通道(零耦合似乎有點(diǎn)過激)。現(xiàn)在邏輯模塊通知UI模塊,只需要定義新的事件類型,在
兩邊分別寫通知和處理通知的代碼即可。

PS:
針對(duì)一些評(píng)論,我再解釋下。其實(shí)any只是用于包裝Param列表而已,這里也可以用void*,再轉(zhuǎn)成
Param*。在這里過多地關(guān)注是用any*還是用void*其實(shí)偏離了本文的重點(diǎn)。本文的重點(diǎn)其實(shí)是Param:

 

OnNotify( NT_ENTER_RGN, ang( create_param( player, rgn_id ) ) );

->
void OnNotify( long type, any data )
{
 Param2
<Player*long> ParamType;
 ParamType 
*= any_cast<ParamType>&data );
 Player 
*player = p->p1;
 
long rgn_id = p->p2;
}




下載相關(guān)代碼

posted on 2010-04-20 22:21 avatar 閱讀(475) 評(píng)論(1)  編輯 收藏 引用 所屬分類: 游戲開發(fā)

評(píng)論

# re: [轉(zhuǎn)載]低耦合模塊間的通信組件:兩個(gè)模板 2010-04-21 16:31 Kevin Lynx

不明真相地路過。。。  回復(fù)  更多評(píng)論   

<2010年12月>
2829301234
567891011
12131415161718
19202122232425
2627282930311
2345678

導(dǎo)航

統(tǒng)計(jì)

常用鏈接

留言簿(2)

隨筆分類

隨筆檔案

搜索

最新評(píng)論

閱讀排行榜

評(píng)論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            在线视频欧美日韩| 开元免费观看欧美电视剧网站| 亚洲免费成人| 欧美一区二区视频免费观看| 免费在线看一区| 国产精品男人爽免费视频1| 亚洲国产美女久久久久| 午夜精品久久久久久久99黑人| 欧美风情在线| 久久精品国产一区二区三区免费看| 欧美日韩另类一区| 亚洲国内欧美| 一区二区激情视频| 欧美精品自拍| 亚洲精品综合| 亚洲国产精品ⅴa在线观看| 欧美一区二区三区婷婷月色 | 欧美大片在线观看| 在线激情影院一区| 麻豆精品视频在线观看| 久久er精品视频| 国内免费精品永久在线视频| 欧美一区二区免费| 久久免费黄色| 亚洲欧洲一区二区三区| 欧美大片免费看| 欧美视频中文在线看 | 国产精品xxxav免费视频| 99精品欧美一区二区三区综合在线| 免费观看不卡av| 美女精品一区| 亚洲毛片网站| 欧美综合国产| 亚洲专区在线| 午夜久久久久久| 黄色精品在线看| 欧美jizzhd精品欧美喷水 | 亚洲国产精品尤物yw在线观看| 国产精品福利在线观看网址| 香港久久久电影| 性久久久久久久久| 亚洲私人黄色宅男| 性欧美videos另类喷潮| 国产精品99久久久久久久久久久久| 久久精品在线观看| 羞羞视频在线观看欧美| 欧美日韩高清不卡| 久久精品九九| 欧美肥婆bbw| 午夜性色一区二区三区免费视频| 免费日韩成人| 亚洲第一精品影视| 亚洲精品美女在线| 亚洲综合二区| 一区在线观看| 亚洲看片免费| 国产一区二区三区黄| 亚洲电影在线免费观看| 黄色一区二区在线| 久久久精品tv| 老鸭窝91久久精品色噜噜导演| 国产视频亚洲精品| 亚洲啪啪91| 国产拍揄自揄精品视频麻豆| 欧美chengren| 亚洲国产高清一区二区三区| 亚洲图片在线| 亚洲经典在线看| 欧美成人蜜桃| 亚洲欧洲一区二区三区久久| 亚洲精品一二区| 久久久久国产一区二区三区| 亚洲视频电影在线| 国产精品久久久久久超碰| 亚洲视频免费在线| 久久精品国产精品 | 欧美日本在线一区| 99精品视频免费观看视频| 伊伊综合在线| 午夜视频精品| 免费在线成人av| 日韩午夜在线观看视频| 欧美少妇一区二区| 午夜在线视频一区二区区别| 久久夜精品va视频免费观看| 国产精品视频免费观看| 亚洲精品男同| 欧美一区二区视频在线观看| 国产在线视频欧美| 欧美激情第二页| 亚洲一品av免费观看| 美女黄网久久| 亚洲网站视频| 欧美大片免费看| 亚洲午夜电影在线观看| 久久视频在线看| 在线一区日本视频| 欧美日韩四区| 久久久久久久一区二区三区| 亚洲激情视频网站| 久久精品一区二区| 国产日韩在线一区| 欧美精品18videos性欧美| 亚洲第一精品久久忘忧草社区| 亚洲天堂av在线免费| 狠狠噜噜久久| 国产精品高潮呻吟久久av无限| 性欧美大战久久久久久久久| 亚洲成色777777女色窝| 一区二区三区在线不卡| 欧美日韩1区| 久久精品女人的天堂av| 在线亚洲观看| 亚洲精品欧美激情| 欧美成人一区二区| 久久国产精品久久久久久电车 | 欧美一区永久视频免费观看| 欧美激情第三页| 亚洲精品一区二区三区樱花| 亚洲韩国日本中文字幕| 一区二区三区av| 亚洲国产色一区| 黄色另类av| 国产精品久久久久久超碰| 欧美精品少妇一区二区三区| 久久久久久9| 久久精品国产亚洲精品 | 制服丝袜激情欧洲亚洲| 亚洲高清资源综合久久精品| 国产一区二区三区四区五区美女| 欧美午夜视频在线| 欧美日韩国产片| 欧美高清你懂得| 欧美高清在线视频| 免费在线观看成人av| 久久午夜精品| 久久精品国产999大香线蕉| 亚洲欧美日韩国产成人| 免费成人美女女| 麻豆av福利av久久av| 玖玖国产精品视频| 免费日韩成人| 欧美激情亚洲自拍| 亚洲国产美女| 日韩网站在线观看| 一二三四社区欧美黄| 在线亚洲观看| 欧美一区二区三区电影在线观看| 亚洲欧美激情四射在线日| 午夜精品久久久久久久男人的天堂| 亚洲综合日韩| 久久九九99| 欧美激情一区在线观看| 欧美日韩一区二区三区免费看| 欧美日韩一级大片网址| 国产精品久久毛片a| 国产日韩在线看| 亚洲高清久久久| 一区二区三区免费观看| 午夜精品福利在线| 久久人人爽人人爽爽久久| 亚洲欧美日韩一区二区三区在线观看 | 欧美国产精品v| 国产精品99免费看| 国产欧美亚洲视频| 亚洲国产精品久久久久久女王| 亚洲精品三级| 欧美在线视频不卡| 欧美激情精品久久久久久久变态| 亚洲欧洲日夜超级视频| 亚洲天堂网在线观看| 久久精品盗摄| 欧美午夜在线观看| 国内精品久久久久久 | 亚洲第一中文字幕| 亚洲一区二区三区高清 | 久久er精品视频| 欧美男人的天堂| 国产综合第一页| 亚洲最新中文字幕| 一区二区三区高清视频在线观看| 亚洲欧美日韩一区| 欧美国产综合| 午夜精品久久久久久久99樱桃| 老司机凹凸av亚洲导航| 国产精品欧美日韩一区二区| 洋洋av久久久久久久一区| 在线观看国产成人av片| 国产一区二区三区观看| 精品99视频| 国产精品99久久久久久久久久久久 | 午夜免费日韩视频| 欧美成黄导航| 国产欧美一区二区精品仙草咪| 亚洲黄色尤物视频| 久久精品一区中文字幕| 日韩一二在线观看| 欧美 日韩 国产一区二区在线视频 | 激情欧美国产欧美| 午夜精品成人在线视频|