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

cyt
VC里面實(shí)現(xiàn)動態(tài)對象載入已經(jīng)不是什么新鮮事情了,很多的plug-in技術(shù)就是例子。Unix下,通過動態(tài)載入so獲得一個對象也不是什么難事,不過對這個對象的管理就是一件比較麻煩的事情了。一般的需求如下:
??有class TMyObj,準(zhǔn)確說TMyObj應(yīng)該是一個接口,根據(jù)不同具體情況會有不同的實(shí)現(xiàn),例如 TMyObj1、TMyObj2等等……而這些TMyObj1和TMyObj2分別保存在不同的so當(dāng)中,需要根據(jù)不同的時候load不同的so,建立相應(yīng)的對象。由于這些對象都擁有TMyObj的接口,所以對于外部來說對這些類的使用就像對TMyObj的使用一樣。
??看起來好像比較簡單,只要在so里面引出一個函數(shù):
??TMyObj * onCreateObject(void);
??而函數(shù)在so中的具體實(shí)現(xiàn)就是建立不同的子類,例如在obj1.so中:
??TMyObj * onCreateObject(void)
???{?return new TMyObj1;?}
??使用的時候只需要動態(tài)load入obj1.so,并且找到onCreateObject函數(shù)的入口,就可以建立一個具有TMyObj接口的TMyObj1了。
??至于釋放對象,一般有兩種方法:
方法一:
??so中包含另外一個函數(shù):
??void onDestroyObj(void * p)
??{
????TMyObj1 * tp = (TMyObj1 *)p;
????delete tp;
??}
??從so中導(dǎo)出該函數(shù),并在刪除對象的時候調(diào)用。
方法二:
??TMyObj的析構(gòu)函數(shù)聲明為虛函數(shù),那么從so導(dǎo)出的onCreateObject()建立的對象,直接執(zhí)行delete刪除就行了,由于析構(gòu)函數(shù)是虛函數(shù),編譯器會正確的調(diào)用TMyObj1的析構(gòu)函數(shù)。
??當(dāng)然,方法二是比較簡單而優(yōu)雅的方法,既然對于C++來說接口就相當(dāng)于純虛函數(shù),多增加一個析構(gòu)的虛函數(shù)又何妨呢。但是無論使用哪種方法,都要注意一個問題,就是載入的obj1.so的生命周期要比最后一個TMyObj1的生存周期長。即只要內(nèi)存中還存在TMyObj1對象,obj1.so就要一直在內(nèi)存中,不能卸載。要保證這個同步,是比較麻煩的事情。下面就說說我的解決方法:
??
??首先,要選擇一個通用的載入so的lib,這個可以參考一下common c++的DSO(在file.h)里面。(不想使用common c++?我也只是說“參考”而已)。這個支持DLL和so,通過成員函數(shù)void *operator[](const char *);獲得指定的symbol的入口。
??其次,就要選擇一個通用的SmartPtr。這個當(dāng)然Loki是首選,Loki的SmartPtr的靈活性比boost的smart_ptr強(qiáng)多了,而且Loki也小巧的多。
??然后就要實(shí)現(xiàn)一個簡單的so的manager,其實(shí)應(yīng)該說是一個動態(tài)object的factory:
??class TObjFactory : protected DSO
??{
??public:
????TObjFactory(void);
????
????void?load(const std::string & strPath);
????void *?createObj(void) const throw (TSOException);
??protected:
????typedef void * (*funcCreate)(void ** p);
????funcCreate??m_pCreator;
??};
??可以想象這個類干些什么:load就是載入相應(yīng)的so,然后獲得so中onCreateObject函數(shù)的入口,并賦給成員m_pCreator。而createObj就是調(diào)用m_pCreator建立對象。不過有所不同的是 m_pCreator所指向的函數(shù)形式是void * funcCreate(void ** p),而多出來void **p用處就是可以讓so中的構(gòu)造函數(shù)中產(chǎn)生的exception能夠傳遞出來。這個不能說不是so的麻煩之處,so中函數(shù)的exception不能被外部捕獲,所以只好這樣子做了。
??現(xiàn)在,關(guān)鍵的地方來了,就是要保證這個TObjFactory的生存周期了。選擇Loki的SmartPtr就能派上用場了。
??Loki的SmartPtr可以自己選定適用的StoragePolicy,這正是我們需要的,參考DefaultSPStorage,可以做我們的TMySOStoragePolicy:
??template
???class TMySOStoragePolicy
???{
????.....
???protected:
????void Destroy()
??????? {
??????? ?delete pointee_;
??????? ?m_pFactory = SmartPtr();
??????? }???
???private:
????SmartPtr?m_pFactory;
????StoredType ??????pointee_;
???};
??顯而易見,這樣做的目的就是要保證釋放指針的時候就減少TObjFactory的引用計(jì)數(shù)。
??好了,現(xiàn)在就是主角了:
??template
??class TDObj : public SmartPtr
??{
??public:
????TDObj(void);
????TDObj(const TDObj & obj);
????.....
????
??protected:
????friend class TDObjManager;
????TDObj(T * p, SmartPtr pManager);
??};
??
??class TDObjManager
??{
??public:
????......
????template
?????static TDObj??createObj(const std::string & strKeyName)
?????{
???????SmartPtr pFactory = getFactoryByName(strKeyName);
???????//這里面可以做很多事情了,例如訪問內(nèi)存,查找相應(yīng)的Factory;或者讀取配置文件、讀入新的so并建立新的Factory。
???????//或者根據(jù)一些淘汰算法,先淘汰內(nèi)存的Factory,然后重新載入新的Factory等等。
???????std::auto_ptr _au( static_cast(pFactory->createObj()) );
???????return TDObj( _au.release(), pFactory);
?????}
??};
??
??以后用起來就簡單多了:
??class TMyObj
??{
??public:
???virtual ~TMyObj(void);
???virtual int func(void) = 0;
??};
??
??TDObj?obj1 = TDObjManager::createObj( "obj1.so") );
??TDObj?obj2 = TDObjManager::createObj( "obj2.so") );
??
??cout << obj1->func() << endl;
??cout << obj2->func() << endl;

??說了這么久,都是主程序的調(diào)用,而so中應(yīng)該如何呢?其實(shí)也很簡單:
??class TMyObj1 : public TMyObj
??{
??public:
????TMyObj1(void);
????~TMyObj1(void);

????static void?onStaticInit(void);
????static void?onStaticDestroy(void);
????static const char *?getVersion(void);
????static const char *?getObjectName(void);
????
????virtual int??func(void);
??};
??
??DECLARE_SO_INTERFACE(TMyObj1);
??
??DECLARE_SO_INTERFACE其實(shí)是一個為了方便編寫程序而定義的宏:
??#define DECLARE_SO_INTERFACE(x)?extern "C" {?\
????void onInstallDLL(void);???\
????void onUninstallDLL(void);???\
????const char * onGetVersion(void);?\
????const char * onObjectName(void);?\
????void *?onCreateObject(void ** ppException);??\
???}; \
???void?onInstallDLL(void)?{ x::onStaticInit(); }????\
???void?onUninstallDLL(void)?{ x::onStaticDestroy(); }??\
???const char * onGetVersion(void) { return x::getVersion(); }??\
???const char * onObjectName(void)?{ return x::getObjectName(); }?\
???void *?onCreateObject(void ** pException)?{?\
????try { \
?????*pException = NULL; x * p = new x();?return (void *)p; \
????}catch(std::exception & e) { \
?????*pException = new std::exception(e); \
?????return NULL; ?\
????}?\
???}
???
??可以看到除了導(dǎo)出onCreateObject函數(shù)以外,還導(dǎo)出了:
??TMyObj1::onStaticInit用于載入so的時候執(zhí)行初始化操作;
??TMyObj1::onStaticDestroy用于卸載so的時候執(zhí)行清理操作;
??TMyObj1::getVersion?獲得對象的版本信息
??TMyObj1::onObjectName 獲得對象名信息等
??可以擴(kuò)展前面的TObjFactory,實(shí)現(xiàn)這些功能。

??同理,我們可以做obj2.so:
??class TMyObj2 : public TMyObj
??{
??public:
???TMyObj2(void);
???~TMyObj2(void);
??
???static void?onStaticInit(void);
???static void?onStaticDestroy(void);
???static const char *?getVersion(void);
???static const char *?getObjectName(void);
??
???virtual int??func(void);
??};
??
??DECLARE_SO_INTERFACE(TMyObj2);
??
??
??另外,一個值得討論的問題是:C++由于沒有反射機(jī)制,所以無法實(shí)現(xiàn)設(shè)值注入和構(gòu)造注入,只能實(shí)現(xiàn)接口注入。不過一般來說也已經(jīng)足夠使用了。

posted on 2005-10-08 14:51 cyt 閱讀(1288) 評論(2)  編輯 收藏 引用
Comments

只有注冊用戶登錄后才能發(fā)表評論。
網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


 
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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精品国产自| 亚洲蜜桃精久久久久久久| 校园春色国产精品| 欧美一区国产一区| 香蕉久久夜色| 狠狠久久亚洲欧美专区| 久久夜色精品国产噜噜av| 久久黄色小说| 日韩午夜一区| 亚洲在线观看免费视频| 在线观看欧美成人| 夜夜精品视频一区二区| 国产伦精品一区二区三区高清| 久久久福利视频| 欧美电影免费观看网站| 亚洲欧美日韩另类精品一区二区三区| 亚洲欧美福利一区二区| 伊人男人综合视频网| 亚洲久色影视| 永久555www成人免费| 亚洲人成在线播放| 国产精品一区二区三区观看| 久久免费偷拍视频| 欧美日本一区二区视频在线观看| 亚洲一区二区黄色| 久久青青草原一区二区| 亚洲一区二区三区成人在线视频精品| 久久不见久久见免费视频1| 亚洲精品视频在线观看网站| 亚洲欧美网站| 亚洲视频在线观看三级| 久久亚洲电影| 欧美一区二区三区在线免费观看| 欧美va天堂va视频va在线| 久久精品91| 国产精品乱码一区二三区小蝌蚪| 欧美va亚洲va日韩∨a综合色| 国产精品乱码妇女bbbb| 亚洲国产老妈| 亚洲国产欧美一区二区三区久久| 亚洲欧美日韩视频一区| 中文一区在线| 欧美极品在线视频| 麻豆av福利av久久av| 国产精品视频xxxx| 99精品欧美一区二区蜜桃免费| 亚洲福利视频一区| 久久精品中文字幕免费mv| 亚洲欧洲av一区二区| 欧美黄色大片网站| 欧美黄色网络| 在线看日韩欧美| 久久亚洲一区| 欧美成年人视频网站| 国产综合欧美| 久久国产精品亚洲77777| 午夜国产不卡在线观看视频| 欧美天天在线| 最新国产成人在线观看| 欧美人成在线| 欧美高清在线观看| 激情欧美日韩| 久久蜜臀精品av| 久久综合电影一区| 精品不卡一区| 久久精品欧洲| 欧美激情国产日韩精品一区18| 久久综合五月天婷婷伊人| 1000精品久久久久久久久| 久久xxxx| 欧美电影电视剧在线观看| 激情国产一区二区| 久久久免费精品| 欧美激情1区| a91a精品视频在线观看| 欧美午夜美女看片| 亚洲中无吗在线| 久久久久久久久久久久久久一区 | 亚洲在线中文字幕| 久久精品国产99国产精品澳门| 国产亚洲va综合人人澡精品| 欧美一区二区三区久久精品| 久久午夜视频| 欧美日韩一卡| 久久蜜臀精品av| 伊人久久综合97精品| 欧美精品在线播放| 亚洲人成人77777线观看| 日韩视频三区| 国产精品日韩高清| 欧美在线综合视频| 欧美国产另类| 一本色道久久| 国产视频亚洲| 女主播福利一区| 亚洲性图久久| 免费精品视频| 亚洲永久免费av| 在线国产精品播放| 欧美日韩国产精品一卡| 午夜精品久久久久久久久 | 国产视频久久久久久久| 久久免费精品视频| 在线一区二区三区四区五区| 久久婷婷亚洲| 亚洲欧美另类在线| 亚洲国产精品成人va在线观看| 欧美三级韩国三级日本三斤| 久久国产精品99国产| 99精品视频免费观看视频| 久久综合九色综合网站| 亚洲一区二区三区免费在线观看 | 欧美精品播放| 久久精品观看| 欧美日韩一区二区视频在线观看| 久久久午夜电影| 国产精品一区二区久久国产| 久久久999国产| 中日韩男男gay无套| 久久午夜精品一区二区| 一区二区三区视频在线| 亚洲黄色影院| 伊人夜夜躁av伊人久久| 国产精品丝袜久久久久久app| 欧美精品在线免费播放| 免费不卡欧美自拍视频| 久久国产成人| 久久av一区二区| 亚洲永久精品大片| 中国亚洲黄色| 亚洲桃色在线一区| 一区二区三区久久网| 日韩视频中午一区| 亚洲日韩欧美视频一区| 欧美国产日本高清在线| 欧美不卡一区| 亚洲国产二区| 亚洲精品三级| 一本一本a久久| 一区二区三区 在线观看视| 99国产精品久久久久久久久久| 亚洲区免费影片| 日韩午夜av在线| 一区二区高清| 亚洲一区免费在线观看| 午夜精品久久久久久久99樱桃 | 亚洲一区二区在线免费观看| 亚洲精品美女久久7777777| 亚洲人妖在线| 在线亚洲免费视频| 亚洲欧美一区二区精品久久久| 欧美一级淫片播放口| 久久久久久久成人| 嫩模写真一区二区三区三州| 亚洲国产精品久久人人爱蜜臀 | 亚洲女爱视频在线| 欧美一级在线视频| 久久精品国产精品亚洲| 麻豆精品在线视频| 欧美日韩亚洲三区| 国产精品有限公司| 伊人影院久久| aa日韩免费精品视频一| 午夜伦理片一区| 免费不卡中文字幕视频| 亚洲欧洲一区二区在线播放| 一区二区三区|亚洲午夜| 香蕉乱码成人久久天堂爱免费 | 亚洲国产cao| 中国日韩欧美久久久久久久久| 午夜精品福利在线观看| 久色成人在线| 一区二区激情视频| 久久久精品五月天| 国产精品白丝av嫩草影院| 国语自产精品视频在线看| 亚洲美女啪啪| 久久精品综合一区| 亚洲精品在线三区| 亚洲欧洲免费视频| 欧美一区二区视频97| 欧美成人午夜激情在线| 国产精品一区二区你懂得| 亚洲高清视频在线| 亚洲欧美一区二区激情| 91久久夜色精品国产网站| 香蕉国产精品偷在线观看不卡 | 欧美成人一区二区| 国产亚洲欧美一区在线观看| 日韩视频不卡| 欧美国产三级| 欧美在线不卡| 国产精品乱码一区二三区小蝌蚪| 亚洲精选久久| 欧美高潮视频| 久久人人看视频| 国产日韩在线视频| 亚洲尤物在线视频观看| 亚洲激情社区| 欧美成人精品影院|