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

[轉載]低耦合模塊間的通信組件:兩個模板

用途

在一個UI與邏輯模塊交互比較多的程序中,因為并不想讓兩個模塊發生太大的耦合,基本目標是
可以完全不改代碼地換一個UI。邏輯模塊需要在產生一些事件后通知到UI模塊,并且在這個通知
里攜帶足夠多的信息(數據)給接收通知的模塊,例如UI模塊。邏輯模塊還可能被放置于與UI模
塊不同的線程里。

最初的結構

最開始我直接采用最簡單的方法,邏輯模塊保存一個UI模塊傳過來的listener。當有事件發生時,
就回調相應的接口將此通知傳出去。大致結構如下:

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

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

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

 

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

改進

于是我決定將各種事件通知統一化:

 

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

 

這樣,邏輯模塊只需要創建事件結構,兩個模塊間的通信就只需要一個接口即可:

void OnNotify( const Event &event );

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

void OnNotify( long event_type, ... )

其實,我需要的就是一個可以表面上類型一樣,但其內部保存的數據卻多樣的東西。這樣一想,
模塊就能讓事情簡單化:

 

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 ) );

 

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

幸虧我以前寫過宏遞歸產生代碼的東西,可以自動地生成這種情況下諸如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的版本。其實這樣定義了Param1、Param2的東西之后,又使得OnNotify
的參數不是特定的了。雖然可以把Param也泛化,但是在邏輯層寫過多的模板代碼,總感覺不好。

于是又想到以前寫的一個東西,可以把各種類型包裝成一種類型---對于外界而言:any。any在
boost中有提到,我只是實現了個簡單的版本。any的大致實現手法就是在內部通過多態機制將各
種類型在某種程度上隱藏,如:

 

        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類通過一個base_type類,利用C++多態機制即可將類型隱藏于var_holder里。那么,
最終的事件通知接口成為下面的樣子:

void OnNotify( long type, any data );

OnNotify( ET_ENTER_RGN, any( create_param( player, rgn_id ) ) );其中,create_param
是一個輔助函數,用于創建各種Param對象。

事實上,實現各種ParamN版本,讓其名字不一樣其實有點不妥。還有一種方法可以讓Param的名字
只有一個,那就是模板偏特化。例如:

 

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)>

 

這種方法主要是通過組合出一種函數類型,來實現偏特化。因為我覺得構造一個函數類型給主模版,
并不是一種合情理的事情。但是,即使使用偏特化來讓Param名字看起來只有一個,但對于不同的
實例化版本,還是不同的類型,所以還是需要any來包裝。

實際使用

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

PS:
針對一些評論,我再解釋下。其實any只是用于包裝Param列表而已,這里也可以用void*,再轉成
Param*。在這里過多地關注是用any*還是用void*其實偏離了本文的重點。本文的重點其實是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;
}




下載相關代碼

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

評論

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

不明真相地路過。。。  回復  更多評論   

<2010年4月>
28293031123
45678910
11121314151617
18192021222324
2526272829301
2345678

導航

統計

常用鏈接

留言簿(2)

隨筆分類

隨筆檔案

搜索

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美一区国产二区| 99riav久久精品riav| 欧美一区二区三区在线看| 亚洲一区二区精品| 国产色综合网| 免费看成人av| 欧美国产亚洲精品久久久8v| 一区二区三区欧美成人| 99视频超级精品| 亚洲一区国产| 国产有码一区二区| 亚洲国产片色| 欧美色另类天堂2015| 午夜精品久久久久久久久久久久久| 亚洲一区二区在线看| 激情久久久久久| 亚洲精品久久久久久一区二区| 欧美性淫爽ww久久久久无| 久久久亚洲国产美女国产盗摄| 欧美大片在线观看一区| 欧美一区成人| 欧美成人69| 久久不射2019中文字幕| 欧美大片91| 久久亚洲免费| 国产精品福利片| 欧美激情 亚洲a∨综合| 欧美体内谢she精2性欧美| 久久一区二区精品| 国产精品久久九九| 欧美黄色大片网站| 国产综合精品| 国产精品99久久久久久久久久久久 | 销魂美女一区二区三区视频在线| 久久国产精品免费一区| 亚洲一区二区精品视频| 老妇喷水一区二区三区| 久久gogo国模裸体人体| 欧美日韩亚洲一区二区三区| 欧美成人一区二区三区在线观看| 国产精品久久久久久亚洲调教| 亚洲成人资源网| 国色天香一区二区| 亚洲性视频h| 亚洲特色特黄| 欧美理论视频| 欧美国产欧美亚洲国产日韩mv天天看完整 | 亚洲欧美日韩中文视频| 在线一区二区视频| 欧美精品1区| 欧美激情精品久久久久久免费印度 | 香蕉国产精品偷在线观看不卡| 亚洲视频精选| 欧美日韩亚洲国产一区| 亚洲日韩欧美视频一区| 亚洲精品专区| 欧美激情在线观看| 亚洲国产婷婷综合在线精品| 亚洲国产欧美一区二区三区久久| 久久成人人人人精品欧| 久久久久久97三级| 国产日韩欧美制服另类| 欧美亚洲综合另类| 久久精品亚洲精品国产欧美kt∨| 国产精品专区h在线观看| 亚洲欧美福利一区二区| 欧美有码视频| 免费日韩成人| 久久国产精品一区二区三区四区| 欧美日韩国产高清视频| 亚洲人成毛片在线播放| 亚洲乱码一区二区| 欧美日韩国产一区二区三区| 91久久久久久久久| 亚洲一区二区三区影院| 国产精品老女人精品视频| 亚洲欧美精品在线观看| 久久精品视频免费观看| 亚洲电影网站| 欧美日韩国产综合久久| 在线视频欧美日韩| 久久久久久久久一区二区| 在线看欧美日韩| 欧美大片一区二区| 一区二区三区四区蜜桃| 欧美伊人影院| 亚洲成色999久久网站| 免费成人在线观看视频| 一本高清dvd不卡在线观看| 亚洲欧美日韩系列| 国产欧美日韩在线视频| 久久免费精品视频| 亚洲精选中文字幕| 久久久综合精品| 亚洲每日更新| 国产专区欧美精品| 欧美精品自拍偷拍动漫精品| 亚洲午夜久久久久久久久电影院 | 亚洲欧美电影院| 亚洲承认在线| 国产精品久久久久久久久久直播 | 国产精品美女久久久久久2018| 亚洲欧美综合一区| 亚洲丰满在线| 久久精品毛片| 一区二区三区欧美在线观看| 国内精品久久久久久久影视蜜臀| 欧美激情bt| 久久久久久久综合| 日韩视频在线一区二区三区| 久久精品中文字幕一区| 亚洲精品黄色| 狠狠色丁香久久婷婷综合丁香 | 在线视频日本亚洲性| 麻豆成人小视频| 午夜久久资源| 在线视频日本亚洲性| 一色屋精品亚洲香蕉网站| 欧美日韩亚洲高清一区二区| 久久久青草婷婷精品综合日韩| 这里只有精品丝袜| 亚洲片在线资源| 欧美激情久久久久| 久久激情一区| 亚洲午夜在线视频| 一区二区三区久久精品| 亚洲欧洲精品一区二区三区波多野1战4| 国产精品中文字幕在线观看| 国产精品第2页| 欧美少妇一区二区| 欧美日韩精品伦理作品在线免费观看| 亚洲一区二区三区精品在线| 欧美图区在线视频| 久久噜噜噜精品国产亚洲综合| 亚洲男人天堂2024| 亚洲一区日韩在线| 亚洲一二三四久久| 亚洲一级网站| 亚洲欧美福利一区二区| 亚洲综合国产| 午夜亚洲一区| 久久国产婷婷国产香蕉| 欧美综合激情网| 久久久精品日韩| 久久亚洲综合| 欧美成熟视频| 欧美日韩成人网| 欧美午夜视频网站| 国产精品视频一区二区三区| 国产精品三区www17con| 国产午夜久久久久| 一区在线观看视频| 亚洲高清不卡在线| 99www免费人成精品| 亚洲男女自偷自拍| 欧美在线观看视频| 葵司免费一区二区三区四区五区| 久久亚洲精品一区| 亚洲激情在线观看| 亚洲视频二区| 久久久久久久综合色一本| 久久这里有精品15一区二区三区| 欧美va亚洲va国产综合| 欧美日韩你懂的| 国产欧美日韩三区| 亚洲国产精品成人| 亚洲午夜影视影院在线观看| 久久国产精品亚洲77777| 欧美成人精品高清在线播放| 亚洲精品乱码久久久久久日本蜜臀| 在线亚洲免费视频| 久久午夜视频| 欧美午夜精品久久久久免费视| 国产日韩欧美一区二区三区在线观看 | 奶水喷射视频一区| 国产精品v亚洲精品v日韩精品| 国产欧美日韩视频在线观看| 91久久视频| 欧美在线综合| 亚洲精品一区二| 久久精品国产亚洲高清剧情介绍| 欧美激情在线免费观看| 国产九九精品| 亚洲福利精品| 欧美一区二区三区免费视频| 亚洲国产激情| 久久成人这里只有精品| 欧美日韩精品久久久| 在线观看欧美成人| 午夜在线一区| 亚洲精品久久久久中文字幕欢迎你 | 欧美激情第一页xxx| 国产精品手机视频| 99精品国产一区二区青青牛奶| 欧美综合国产精品久久丁香| 亚洲精品视频中文字幕| 免费91麻豆精品国产自产在线观看| 国产精品一区二区黑丝| 国产精品99久久久久久www| 欧美高清在线一区|