C++中的RTTI機(jī)制詳解
前言
RTTI是”Runtime Type Information”的縮寫,意思是運(yùn)行時類型信息,它提供了運(yùn)行時確定對象類型的方法。RTTI并不是什么新的東西,很早就有了這個技術(shù),但是,在實(shí)際應(yīng)用中使用的比較少而已。而我這里就是對RTTI進(jìn)行總結(jié),今天我沒有用到,并不代表這個東西沒用。學(xué)無止境,先從typeid函數(shù)開始講起。
typeid函數(shù)
typeid的主要作用就是讓用戶知道當(dāng)前的變量是什么類型的,比如以下代碼:
#include <iostream>
#include <typeinfo>
using namespace std;
int main()
{
short s = 2;
unsigned ui = 10;
int i = 10;
char ch = 'a';
wchar_t wch = L'b';
float f = 1.0f;
double d = 2;
cout<<typeid(s).name()<<endl; // short
cout<<typeid(ui).name()<<endl; // unsigned int
cout<<typeid(i).name()<<endl; // int
cout<<typeid(ch).name()<<endl; // char
cout<<typeid(wch).name()<<endl; // wchar_t
cout<<typeid(f).name()<<endl; // float
cout<<typeid(d).name()<<endl; // double
return 0;
}
對于C++支持的內(nèi)建類型,typeid能完全支持,我們通過調(diào)用typeid函數(shù),我們就能知道變量的信息。對于我們自定義的結(jié)構(gòu)體,類呢?
#include <iostream>
#include <typeinfo>
using namespace std;
class A
{
public:
void Print() { cout<<"This is class A."<<endl; }
};
class B : public A
{
public:
void Print() { cout<<"This is class B."<<endl; }
};
struct C
{
void Print() { cout<<"This is struct C."<<endl; }
};
int main()
{
A *pA1 = new A();
A a2;
cout<<typeid(pA1).name()<<endl; // class A *
cout<<typeid(a2).name()<<endl; // class A
B *pB1 = new B();
cout<<typeid(pB1).name()<<endl; // class B *
C *pC1 = new C();
C c2;
cout<<typeid(pC1).name()<<endl; // struct C *
cout<<typeid(c2).name()<<endl; // struct C
return 0;
}
是的,對于我們自定義的結(jié)構(gòu)體和類,tpyeid都能支持。在上面的代碼中,在調(diào)用完typeid之后,都會接著調(diào)用name()函數(shù),可以看出typeid函數(shù)返回的是一個結(jié)構(gòu)體或者類,然后,再調(diào)用這個返回的結(jié)構(gòu)體或類的name成員函數(shù);其實(shí),typeid是一個返回類型為type_info類型的函數(shù)。那么,我們就有必要對這個type_info類進(jìn)行總結(jié)一下,畢竟它實(shí)際上存放著類型信息。
type_info類
去掉那些該死的宏,在Visual Studio 2012中查看type_info類的定義如下:
class type_info
{
public:
virtual ~type_info();
bool operator==(const type_info& _Rhs) const; // 用于比較兩個對象的類型是否相等
bool operator!=(const type_info& _Rhs) const; // 用于比較兩個對象的類型是否不相等
bool before(const type_info& _Rhs) const;
// 返回對象的類型名字,這個函數(shù)用的很多
const char* name(__type_info_node* __ptype_info_node = &__type_info_root_node) const;
const char* raw_name() const;
private:
void *_M_data;
char _M_d_name[1];
type_info(const type_info& _Rhs);
type_info& operator=(const type_info& _Rhs);
static const char * _Name_base(const type_info *,__type_info_node* __ptype_info_node);
static void _Type_info_dtor(type_info *);
};
在type_info類中,復(fù)制構(gòu)造函數(shù)和賦值運(yùn)算符都是私有的,同時也沒有默認(rèn)的構(gòu)造函數(shù);所以,我們沒有辦法創(chuàng)建type_info類的變量,例如type_info A;這樣是錯誤的。那么typeid函數(shù)是如何返回一個type_info類的對象的引用的呢?我在這里不進(jìn)行討論,思路就是類的友元函數(shù)。
typeid函數(shù)的使用
typeid使用起來是非常簡單的,常用的方式有以下兩種:
1.使用type_info類中的name()函數(shù)返回對象的類型名稱
就像上面的代碼中使用的那樣;但是,這里有一點(diǎn)需要注意,比如有以下代碼:
#include <iostream>
#include <typeinfo>
using namespace std;
class A
{
public:
void Print() { cout<<"This is class A."<<endl; }
};
class B : public A
{
public:
void Print() { cout<<"This is class B."<<endl; }
};
int main()
{
A *pA = new B();
cout<<typeid(pA).name()<<endl; // class A *
cout<<typeid(*pA).name()<<endl; // class A
return 0;
}
我使用了兩次typeid,但是兩次的參數(shù)是不一樣的;輸出結(jié)果也是不一樣的;當(dāng)我指定為pA時,由于pA是一個A類型的指針,所以輸出就為class A *;當(dāng)我指定*pA時,它表示的是pA所指向的對象的類型,所以輸出的是class A;所以需要區(qū)分typeid(*pA)和typeid(pA)的區(qū)別,它們兩個不是同一個東西;但是,這里又有問題了,明明pA實(shí)際指向的是B,為什么得到的卻是class A呢?我們在看下一段代碼:
#include <iostream>
#include <typeinfo>
using namespace std;
class A
{
public:
virtual void Print() { cout<<"This is class A."<<endl; }
};
class B : public A
{
public:
void Print() { cout<<"This is class B."<<endl; }
};
int main()
{
A *pA = new B();
cout<<typeid(pA).name()<<endl; // class A *
cout<<typeid(*pA).name()<<endl; // class B
return 0;
}
好了,我將Print函數(shù)變成了虛函數(shù),輸出結(jié)果就不一樣了,這說明什么?這就是RTTI在搗鬼了,當(dāng)類中不存在虛函數(shù)時,typeid是編譯時期的事情,也就是靜態(tài)類型,就如上面的cout<<typeid(*pA).name()<<endl;輸出class A一樣;當(dāng)類中存在虛函數(shù)時,typeid是運(yùn)行時期的事情,也就是動態(tài)類型,就如上面的cout<<typeid(*pA).name()<<endl;輸出class B一樣,關(guān)于這一點(diǎn),我們在實(shí)際編程中,經(jīng)常會出錯,一定要謹(jǐn)記。
2.使用type_info類中重載的==和!=比較兩個對象的類型是否相等
這個會經(jīng)常用到,通常用于比較兩個帶有虛函數(shù)的類的對象是否相等,例如以下代碼:
#include <iostream>
#include <typeinfo>
using namespace std;
class A
{
public:
virtual void Print() { cout<<"This is class A."<<endl; }
};
class B : public A
{
public:
void Print() { cout<<"This is class B."<<endl; }
};
class C : public A
{
public:
void Print() { cout<<"This is class C."<<endl; }
};
void Handle(A *a)
{
if (typeid(*a) == typeid(A))
{
cout<<"I am a A truly."<<endl;
}
else if (typeid(*a) == typeid(B))
{
cout<<"I am a B truly."<<endl;
}
else if (typeid(*a) == typeid(C))
{
cout<<"I am a C truly."<<endl;
}
else
{
cout<<"I am alone."<<endl;
}
}
int main()
{
A *pA = new B();
Handle(pA);
delete pA;
pA = new C();
Handle(pA);
return 0;
}
這是一種用法,呆會我再總結(jié)如何使用dynamic_cast來實(shí)現(xiàn)同樣的功能。
dynamic_cast的內(nèi)幕
在這篇《static_cast、dynamic_cast、const_cast和reinterpret_cast總結(jié)》的文章中,也介紹了dynamic_cast的使用,對于dynamic_cast到底是如何實(shí)現(xiàn)的,并沒有進(jìn)行說明,而這里就要對于dynamic_cast的內(nèi)幕一探究竟。首先來看一段代碼:
#include <iostream>
#include <typeinfo>
using namespace std;
class A
{
public:
virtual void Print() { cout<<"This is class A."<<endl; }
};
class B
{
public:
virtual void Print() { cout<<"This is class B."<<endl; }
};
class C : public A, public B
{
public:
void Print() { cout<<"This is class C."<<endl; }
};
int main()
{
A *pA = new C;
//C *pC = pA; // Wrong
C *pC = dynamic_cast<C *>(pA);
if (pC != NULL)
{
pC->Print();
}
delete pA;
}
在上面代碼中,如果我們直接將pA賦值給pC,這樣編譯器就會提示錯誤,而當(dāng)我們加上了dynamic_cast之后,一切就ok了。那么dynamic_cast在后面干了什么呢?
dynamic_cast主要用于在多態(tài)的時候,它允許在運(yùn)行時刻進(jìn)行類型轉(zhuǎn)換,從而使程序能夠在一個類層次結(jié)構(gòu)中安全地轉(zhuǎn)換類型,把基類指針(引用)轉(zhuǎn)換為派生類指針(引用)。我在《COM編程——接口的背后》這篇博文中總結(jié)的那樣,當(dāng)類中存在虛函數(shù)時,編譯器就會在類的成員變量中添加一個指向虛函數(shù)表的vptr指針,每一個class所關(guān)聯(lián)的type_info object也經(jīng)由virtual table被指出來,通常這個type_info object放在表格的第一個slot。當(dāng)我們進(jìn)行dynamic_cast時,編譯器會幫我們進(jìn)行語法檢查。如果指針的靜態(tài)類型和目標(biāo)類型相同,那么就什么事情都不做;否則,首先對指針進(jìn)行調(diào)整,使得它指向vftable,并將其和調(diào)整之后的指針、調(diào)整的偏移量、靜態(tài)類型以及目標(biāo)類型傳遞給內(nèi)部函數(shù)。其中最后一個參數(shù)指明轉(zhuǎn)換的是指針還是引用。兩者唯一的區(qū)別是,如果轉(zhuǎn)換失敗,前者返回NULL,后者拋出bad_cast異常。對于在typeid函數(shù)的使用中所示例的程序,我使用dynamic_cast進(jìn)行更改,代碼如下:
#include <iostream>
#include <typeinfo>
using namespace std;
class A
{
public:
virtual void Print() { cout<<"This is class A."<<endl; }
};
class B : public A
{
public:
void Print() { cout<<"This is class B."<<endl; }
};
class C : public A
{
public:
void Print() { cout<<"This is class C."<<endl; }
};
void Handle(A *a)
{
if (dynamic_cast<B*>(a))
{
cout<<"I am a B truly."<<endl;
}
else if (dynamic_cast<C*>(a))
{
cout<<"I am a C truly."<<endl;
}
else
{
cout<<"I am alone."<<endl;
}
}
int main()
{
A *pA = new B();
Handle(pA);
delete pA;
pA = new C();
Handle(pA);
return 0;
}
這個是使用dynamic_cast進(jìn)行改寫的版本。實(shí)際項目中,這種方法會使用的更多點(diǎn)。
總結(jié)
我在這里總結(jié)了RTTI的相關(guān)知識,希望大家看懂了。這篇博文有點(diǎn)長,希望大家也耐心的看??偨Y(jié)了就會有收獲。
欄 目:C語言
下一篇:C++基于CreateToolhelp32Snapshot獲取系統(tǒng)進(jìn)程實(shí)例
本文標(biāo)題:C++中的RTTI機(jī)制詳解
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/3295.html
您可能感興趣的文章
- 04-02c語言沒有round函數(shù) round c語言
- 01-10深入理解C++中常見的關(guān)鍵字含義
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10深入Main函數(shù)中的參數(shù)argc,argv的使用詳解
- 01-10c++中inline的用法分析
- 01-10如何尋找數(shù)組中的第二大數(shù)
- 01-10用C++實(shí)現(xiàn)DBSCAN聚類算法
- 01-10全排列算法的非遞歸實(shí)現(xiàn)與遞歸實(shí)現(xiàn)的方法(C++)
- 01-10C++大數(shù)模板(推薦)
- 01-10淺談C/C++中的static與extern關(guān)鍵字的使用詳解


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達(dá)式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 04-02jquery與jsp,用jquery
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-10C#中split用法實(shí)例總結(jié)
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10delphi制作wav文件的方法
- 08-05織夢dedecms什么時候用欄目交叉功能?
- 01-10使用C語言求解撲克牌的順子及n個骰子
- 01-11ajax實(shí)現(xiàn)頁面的局部加載