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

牧光小院

被約束的日日夜夜,停不下來(lái)的時(shí)間。

MFC漫談(一)——RTTI

所有的這一系列的東西都來(lái)源于前天晚上的一個(gè)電話,內(nèi)容大概是說(shuō): 你能教會(huì)我一個(gè)讓我對(duì) MFC 有點(diǎn)感覺(jué)的 Hello World 嗎?我渴望一個(gè)像用 C 寫(xiě)的 Win32 Hello World 一樣直觀的例子。 想想這曾經(jīng)是我學(xué)習(xí) MFC 的時(shí)候也想到過(guò)的問(wèn)題。我是一個(gè)喜歡刨根問(wèn)底的人,喜歡把事情搞明白,于是曾經(jīng)很長(zhǎng)的一段時(shí)間里,我都困惑在紛繁雜亂的代碼里?,F(xiàn)在回想起來(lái),侯捷老師的《 Dessecting MFC 》和 Jeff Prosise 的《 Programming MFC 》一起讀來(lái),估計(jì)能達(dá)到解惑的目的。當(dāng)然,需要的是一點(diǎn)點(diǎn)耐心和對(duì) Win32 程序的一點(diǎn)最基本的了解(貌似廢話。。。)。于是那天晚上, QQ 上和那個(gè)朋友聊了挺長(zhǎng)一段時(shí)間,翻騰了一下 MFC 的源代碼,就有了這一系列的東西,全當(dāng)是故地重游了一番。言歸正傳吧。文章中所有的代碼都提取自MFC 7.0


在MFC中,RTTI是依靠為彼此有繼承關(guān)系的類建立一個(gè)記錄其類型的鏈表來(lái)實(shí)現(xiàn)的,和RTTI有關(guān)的CRuntimeClass成員有4個(gè):
LPCSTR?m_lpszClassName;????//?用于記錄類名
//?用于指向基類的CRuntimeClass結(jié)構(gòu)
CRuntimeClass*?m_pBaseClass;??
//?用于指向鏈表中前一個(gè)類的CRuntimeClass結(jié)構(gòu)
CRuntimeClass*?m_pNextClass;
//?用于建立類別型錄
const?AFX_CLASSINIT*?m_pClassInit;?

這樣在這個(gè)類別型錄中就有了許多條路徑,每一條都是沿著m_pBaseClass一直可以找到某個(gè)類的最終基類。要把一個(gè)類加入到這個(gè)類別型錄中要用到兩個(gè)宏:
DECLARE_DYNAMIC?/?IMPLEMENT_DYNAMIC
其中:
#define?DECLARE_DYNAMIC?(class_name)?\
public:?\
????
static?const?CRuntimeClass?class##class_name;?\
????
virtual?CRuntimeClass*?GetRuntimeClass()?const;?\

這個(gè)宏是用在類聲明中的,其作用就是根據(jù)類的名字為該類添加兩個(gè)public的成員,分別用于記錄類的型別和獲得對(duì)象class##class_name的地址,注意這里的class##class_name是個(gè)靜態(tài)成員,這就為后面我們做類型的比較奠定了基礎(chǔ)(繼承于同一個(gè)基類的派生類對(duì)象包含共同的靜態(tài)類成員)。在類中使用了DECLARE_DYNAMIC后,還要在.cpp的文件中使用IMPLEMENT_DYNAMIC宏,該宏的作用就是初始化class##class_name對(duì)象和定義GetRuntimeClass函數(shù)。
#define?IMPLEMENT_DYNAMIC?(class_name,?base_class_name)?\
????IMPLEMENT_RUNTIMECLASS?(class_name,?base_class_name,?
0xFFFF,?NULL,?NULL)

IMPLEMENT_DYNAMIC在使用的時(shí)候,要指定類和其基類的名字,之后利用IMPLEMENT_RUNTIMECLASS進(jìn)行實(shí)質(zhì)性的初始化活動(dòng)。
#define?IMPLEMENT_RUNTIMECLASS?(class_name,?base_class_name,?wSchema,?pfnNew,?class_init)?\
????AFX_COMDAT?
const?CRuntimeClass?class_name::class##class_name?=?{?\
????????  #class_name,?
sizeof(class?class_name),?wSchema,?pfnNew,?\
????????  RUNTIME_CLASS(base_class_name),?NULL,?class_init?}
;?\

????CRuntimeClass
*?class_name::GetRuntimeClass()?const?\
????
{?return?RUNTIME_CLASS?(class_name);?}?\

其中,在class#class_name的初始化中和RTTI相關(guān)的只有:
&name_class用來(lái)初始化m_lpszClassName
RUNTIME_CLASS(base_class_name)用來(lái)初始化CRuntimeClass* m_pBaseClass
NULL用來(lái)初始化CRuntimeClass* m_pNextClass(此時(shí)類別型錄還沒(méi)有建立起來(lái))

另外,RUNTIME_CLASS就是用來(lái)獲得class##class_name對(duì)象地址的宏:

#define?RUNTIME_CLASS?(class_name)?_RUNTIME_CLASS?(class_name)
#define?_RUNTIME_CLASS?(class_name)?\
????((CRuntimeClass
*)?(&class_name::class##class_name))

這樣,當(dāng)對(duì)程序中的每一個(gè)類都使用了DECLARE_DYNAMIC / IMPLEMENT_DYNAMIC宏之后,就為該類在類別型錄中進(jìn)行了登記工作。當(dāng)然,MFC中所有的類都派生于CObject,所以所有的路線最終都要在CObject處會(huì)合,由于CObject沒(méi)有基類,所以它的CRuntimeClass對(duì)象并不能用上面的兩個(gè)宏來(lái)實(shí)現(xiàn),在objcore.cpp中,為CObject的classCObject對(duì)象單獨(dú)作了初始化的工作:

const?struct?CRuntimeClass?CObject::classCObject?=
?
{?"CObject",?sizeof(CObject),?0xffff,?NULL,?NULL,?NULL?};

我們可以看到m_pBaseClass被初始化為NULL。另外,也單獨(dú)實(shí)現(xiàn)了GetRuntimeClass():

CRuntimeClass*?CObject::GetRuntimeClass()?const?{
????
return?_RUNTIME_CLASS?(CObject);
}

至于_RUNTIME_CLASS,前面已經(jīng)說(shuō)過(guò)了。這樣,如果想要把自己的類介紹給MFC,只要在類聲明中使用DECLARE_DYNAMIC,在類的實(shí)現(xiàn)中加入IMPLEMENT_DYNAMIC
,就可以把自己注冊(cè)到類別型錄中了。至此,為了實(shí)現(xiàn)類對(duì)象的RTTI,我們已經(jīng)做好了所有的準(zhǔn)備工作,下面就來(lái)看一下它的實(shí)現(xiàn),它主要是靠CObject中的IsKindOf函數(shù)完成的。

BOOL?CObject::IsKindOf(const?CRuntimeClass*?pClass)?const
{
?  
//?為了簡(jiǎn)潔,略去了不相關(guān)的代碼
?  CRuntimeClass*?pClassThis?=?GetRuntimeClass();
?  
return?pClassThis->IsDerivedFrom(pClass);
}

這里,由于GetRuntimeClass是虛函數(shù),所以pClassThis會(huì)指向調(diào)用IsKindOf函數(shù)的類對(duì)象的class##class_name,之后利用指向該對(duì)象的指針調(diào)用IsDerivedFrom:

BOOL?CRuntimeClass::IsDerivedFrom(const?CRuntimeClass*?pBaseClass)?const?{
????
//為了簡(jiǎn)潔,略去了不相關(guān)的代碼
????if?(pBaseClass?==?NULL)
????????
return?FALSE;

????
//?simple?SI?case
????const?CRuntimeClass*?pClassThis?=?this;
????
while?(pClassThis?!=?NULL)?{
????????
if?(pClassThis?==?pBaseClass)
???????? 
return?TRUE;
????????pClassThis?
=?pClassThis->m_pBaseClass;
????}
????
????
return?FALSE;???????//?walked?to?the?top,?no?match
}

我們知道,派生類和基類共享基類的static對(duì)象,所以在這里,派生類和基類一定共享相同的class##class_name對(duì)象,這就為我們判定兩個(gè)類是否有繼承關(guān)系提供了理論基礎(chǔ),同樣,在IsDerivedFrom中,while循環(huán)中的if也的確是這樣做的,它沿著該類的同宗路線上行,只要不到共同的祖先CObject,就決不罷休?!?br />
結(jié)論
如果想要把自己的類介紹給MFC,只要在類聲明中使用DECLARE_DYNAMIC,在類的實(shí)現(xiàn)中加入IMPLEMENT_DYNAMIC
,就可以把自己注冊(cè)到類別型錄中了?! 。ùm(xù)……)

posted on 2006-05-18 14:53 nacci 閱讀(4019) 評(píng)論(1)  編輯 收藏 引用 所屬分類: C++漫談

評(píng)論

# re: MFC漫談(一)——RTTI 2006-05-28 19:50 j

看上去好像是侯捷老師的《 Dessecting MFC 》里面的代碼  回復(fù)  更多評(píng)論   

<2025年9月>
31123456
78910111213
14151617181920
21222324252627
2829301234
567891011

導(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>
            女人天堂亚洲aⅴ在线观看| 午夜视频在线观看一区二区| 久久综合激情| 欧美中文字幕| 久久国产精品99国产精| 久久国产免费看| 久久久亚洲国产天美传媒修理工| 久久黄色小说| 男女av一区三区二区色多| 欧美福利视频在线观看| 欧美人与禽猛交乱配视频| 欧美午夜精品久久久久久人妖| 欧美体内she精视频| 国产精品视频免费观看| 国产中文一区二区| 亚洲伦伦在线| 欧美一区1区三区3区公司| 欧美专区中文字幕| 欧美风情在线| 亚洲视频1区| 另类尿喷潮videofree| 99一区二区| 欧美日韩亚洲一区三区| 国产精品看片资源| 一区二区在线不卡| 亚洲一区国产视频| 久久在线免费观看| 亚洲精品国产视频| 亚洲综合欧美| 欧美激情综合网| 国产欧美在线| 一本到12不卡视频在线dvd| 久久久99免费视频| 亚洲美女精品久久| 久久久www成人免费毛片麻豆| 欧美日韩精品在线| 亚洲电影免费| 久久久久久高潮国产精品视| 日韩视频免费在线| 欧美成人网在线| 一区二区亚洲精品| 久久精品免费| 亚洲自拍高清| 国产精品av一区二区| 亚洲美女黄网| 欧美激情亚洲另类| 久久蜜臀精品av| 国产亚洲欧美另类中文| 亚洲欧美日韩精品在线| av成人激情| 欧美少妇一区| 亚洲天堂免费观看| 亚洲人成免费| 欧美精品国产精品| 日韩视频一区二区三区| 欧美激情一区二区久久久| 久久久蜜臀国产一区二区| 国产亚洲a∨片在线观看| 亚洲欧美日韩一区二区三区在线观看| 亚洲人体偷拍| 欧美日韩国产bt| 亚洲视频在线观看免费| 99成人精品| 国产精品久久福利| 欧美一区二区黄| 欧美在线影院| 在线电影国产精品| 欧美jizz19性欧美| 免费在线视频一区| 一区二区三区 在线观看视频| 亚洲黄色一区| 国产精品播放| 久久国产乱子精品免费女 | 一区二区不卡在线视频 午夜欧美不卡在 | 国产精品久久久久影院亚瑟 | 欧美99在线视频观看| 久久人体大胆视频| 亚洲精品视频免费在线观看| 亚洲国产99精品国自产| 欧美国产综合| 一区二区三区免费网站| 99这里只有久久精品视频| 欧美亚州一区二区三区 | 你懂的视频一区二区| 亚洲精品一区二区网址 | 亚洲男女毛片无遮挡| 国产一区二区三区久久久久久久久| 久久精品国产99国产精品澳门| 久久激情视频| 亚洲精选大片| 亚洲欧美日韩人成在线播放| **欧美日韩vr在线| 一区二区激情视频| 黄色综合网站| 一区二区三区免费网站| 黄网站免费久久| 亚洲精品久久久久久久久| 国产精品日韩久久久久| 麻豆九一精品爱看视频在线观看免费| 亚洲国产精品成人综合| 国产精品v日韩精品v欧美精品网站| 久久理论片午夜琪琪电影网| 欧美黄色一级视频| 久久本道综合色狠狠五月| 女人色偷偷aa久久天堂| 欧美一级日韩一级| 欧美日韩1234| 美国十次了思思久久精品导航| 欧美日韩亚洲综合一区| 美女91精品| 国产精品入口福利| 亚洲精品资源| 在线免费观看欧美| 欧美一级欧美一级在线播放| 亚洲毛片在线观看.| 欧美专区一区二区三区| 亚洲欧美日韩视频一区| 欧美另类专区| 亚洲国产成人av在线| 狠狠色综合色区| 香蕉久久国产| 欧美一区在线看| 国产精品久久久久久影视| 日韩视频二区| 一本一道久久综合狠狠老精东影业 | 亚洲视频在线观看免费| 亚洲激情二区| 久久精品夜色噜噜亚洲aⅴ| 性做久久久久久免费观看欧美| 欧美日韩一区在线| 亚洲精品在线一区二区| 亚洲三级毛片| 麻豆精品一区二区av白丝在线| 久久精品免费观看| 国产亚洲毛片| 久久精品99| 免费h精品视频在线播放| 国内精品国语自产拍在线观看| 亚洲欧美日韩一区| 久久福利影视| 伊人久久久大香线蕉综合直播| 久久国产一区二区| 噜噜噜在线观看免费视频日韩 | 欧美另类女人| 日韩视频二区| 亚洲欧美日韩高清| 国产女主播一区二区三区| 亚洲欧美日韩直播| 久久伊人精品天天| 亚洲国产一成人久久精品| 欧美激情一级片一区二区| 亚洲精品一区二区三区婷婷月| 在线亚洲伦理| 国产亚洲va综合人人澡精品| 久久精品一区二区国产| 欧美激情乱人伦| 亚洲摸下面视频| 国产在线精品一区二区中文| 久久天堂国产精品| 亚洲理论在线| 久久网站热最新地址| 亚洲欧洲一区二区在线播放| 欧美日韩午夜视频在线观看| 亚洲综合日本| 欧美成人精品一区二区| 一区二区三区欧美视频| 国产伦精品一区二区三区高清版| 久久成人资源| 99精品国产一区二区青青牛奶| 久久激情五月激情| 999亚洲国产精| 国产有码在线一区二区视频| 欧美大学生性色视频| 亚洲资源在线观看| 亚洲二区视频| 欧美在线一二三| 亚洲人成网站在线播| 国产伦精品一区二区三区| 嫩草伊人久久精品少妇av杨幂| 亚洲自拍偷拍麻豆| 亚洲全黄一级网站| 老司机免费视频一区二区三区| 一本色道88久久加勒比精品| 国产一区二区中文| 欧美日韩午夜| 欧美电影免费观看高清完整版| 亚洲欧美电影在线观看| 日韩天堂在线观看| 亚洲第一页中文字幕| 国模叶桐国产精品一区| 最新成人av在线| 久久久久国产精品厨房| 日韩一区二区电影网| 韩国成人福利片在线播放| 欧美日韩国产在线播放网站| 欧美怡红院视频一区二区三区| 99精品视频免费在线观看| 欧美+亚洲+精品+三区| 久久精品2019中文字幕| 亚洲免费小视频|