亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

C++中的RTTI機制詳解

2020-01-26 15:16:18
字體:
來源:轉載
供稿:網友

前言

RTTI是”Runtime Type Information”的縮寫,意思是運行時類型信息,它提供了運行時確定對象類型的方法。RTTI并不是什么新的東西,很早就有了這個技術,但是,在實際應用中使用的比較少而已。而我這里就是對RTTI進行總結,今天我沒有用到,并不代表這個東西沒用。學無止境,先從typeid函數開始講起。

typeid函數

typeid的主要作用就是讓用戶知道當前的變量是什么類型的,比如以下代碼:

復制代碼 代碼如下:

#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++支持的內建類型,typeid能完全支持,我們通過調用typeid函數,我們就能知道變量的信息。對于我們自定義的結構體,類呢?

復制代碼 代碼如下:

#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;
}

是的,對于我們自定義的結構體和類,tpyeid都能支持。在上面的代碼中,在調用完typeid之后,都會接著調用name()函數,可以看出typeid函數返回的是一個結構體或者類,然后,再調用這個返回的結構體或類的name成員函數;其實,typeid是一個返回類型為type_info類型的函數。那么,我們就有必要對這個type_info類進行總結一下,畢竟它實際上存放著類型信息。

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;
 
    // 返回對象的類型名字,這個函數用的很多
    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類中,復制構造函數和賦值運算符都是私有的,同時也沒有默認的構造函數;所以,我們沒有辦法創建type_info類的變量,例如type_info A;這樣是錯誤的。那么typeid函數是如何返回一個type_info類的對象的引用的呢?我在這里不進行討論,思路就是類的友元函數。

typeid函數的使用

typeid使用起來是非常簡單的,常用的方式有以下兩種:

1.使用type_info類中的name()函數返回對象的類型名稱

就像上面的代碼中使用的那樣;但是,這里有一點需要注意,比如有以下代碼:

復制代碼 代碼如下:

#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,但是兩次的參數是不一樣的;輸出結果也是不一樣的;當我指定為pA時,由于pA是一個A類型的指針,所以輸出就為class A *;當我指定*pA時,它表示的是pA所指向的對象的類型,所以輸出的是class A;所以需要區分typeid(*pA)和typeid(pA)的區別,它們兩個不是同一個東西;但是,這里又有問題了,明明pA實際指向的是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函數變成了虛函數,輸出結果就不一樣了,這說明什么?這就是RTTI在搗鬼了,當類中不存在虛函數時,typeid是編譯時期的事情,也就是靜態類型,就如上面的cout<<typeid(*pA).name()<<endl;輸出class A一樣;當類中存在虛函數時,typeid是運行時期的事情,也就是動態類型,就如上面的cout<<typeid(*pA).name()<<endl;輸出class B一樣,關于這一點,我們在實際編程中,經常會出錯,一定要謹記。

2.使用type_info類中重載的==和!=比較兩個對象的類型是否相等

這個會經常用到,通常用于比較兩個帶有虛函數的類的對象是否相等,例如以下代碼:

復制代碼 代碼如下:

#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;
}

這是一種用法,呆會我再總結如何使用dynamic_cast來實現同樣的功能。

dynamic_cast的內幕

在這篇《static_cast、dynamic_cast、const_cast和reinterpret_cast總結》的文章中,也介紹了dynamic_cast的使用,對于dynamic_cast到底是如何實現的,并沒有進行說明,而這里就要對于dynamic_cast的內幕一探究竟。首先來看一段代碼:

復制代碼 代碼如下:

#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,這樣編譯器就會提示錯誤,而當我們加上了dynamic_cast之后,一切就ok了。那么dynamic_cast在后面干了什么呢?

dynamic_cast主要用于在多態的時候,它允許在運行時刻進行類型轉換,從而使程序能夠在一個類層次結構中安全地轉換類型,把基類指針(引用)轉換為派生類指針(引用)。我在《COM編程――接口的背后》這篇博文中總結的那樣,當類中存在虛函數時,編譯器就會在類的成員變量中添加一個指向虛函數表的vptr指針,每一個class所關聯的type_info object也經由virtual table被指出來,通常這個type_info object放在表格的第一個slot。當我們進行dynamic_cast時,編譯器會幫我們進行語法檢查。如果指針的靜態類型和目標類型相同,那么就什么事情都不做;否則,首先對指針進行調整,使得它指向vftable,并將其和調整之后的指針、調整的偏移量、靜態類型以及目標類型傳遞給內部函數。其中最后一個參數指明轉換的是指針還是引用。兩者唯一的區別是,如果轉換失敗,前者返回NULL,后者拋出bad_cast異常。對于在typeid函數的使用中所示例的程序,我使用dynamic_cast進行更改,代碼如下:

復制代碼 代碼如下:

#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進行改寫的版本。實際項目中,這種方法會使用的更多點。

總結

我在這里總結了RTTI的相關知識,希望大家看懂了。這篇博文有點長,希望大家也耐心的看??偨Y了就會有收獲。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
日韩免费在线观看视频| 色综合久久88色综合天天看泰| 国产美女久久精品| 在线精品播放av| 在线成人免费网站| 日韩亚洲国产中文字幕| 91在线观看免费| 亚洲欧美综合另类中字| 久久99精品久久久久久噜噜| 美女少妇精品视频| 91精品国产高清自在线看超| 久久精品99国产精品酒店日本| 日本欧美一二三区| 亚洲精品国产精品国自产观看浪潮| 欧美高清性猛交| 亚洲高清色综合| 7777免费精品视频| 欧美亚洲另类激情另类| 国产视频在线一区二区| 免费99精品国产自在在线| 午夜精品一区二区三区在线视频| 国产精品青青在线观看爽香蕉| 国产午夜精品理论片a级探花| 4p变态网欧美系列| 伊人亚洲福利一区二区三区| 国产精品美乳一区二区免费| 久久99青青精品免费观看| 97碰碰碰免费色视频| 黄网动漫久久久| 欧美综合国产精品久久丁香| 国产成人拍精品视频午夜网站| 91精品中国老女人| 国产日韩精品视频| 亚洲黄页网在线观看| 成人免费在线网址| 一区二区三区久久精品| 中文字幕久热精品在线视频| 日韩大陆毛片av| 亚洲护士老师的毛茸茸最新章节| 欧美国产第一页| 成人欧美在线视频| 国产一区二区三区在线| 91欧美精品成人综合在线观看| 亚洲人高潮女人毛茸茸| 亚洲成人精品在线| 亚洲国产精品悠悠久久琪琪| 亚洲天堂av电影| 亚洲最大在线视频| 韩国精品久久久999| 国产欧美精品一区二区三区介绍| 久久国产精品亚洲| 日韩有码片在线观看| 91精品视频大全| 韩国19禁主播vip福利视频| 日韩av快播网址| 久久激情视频免费观看| 日本成人精品在线| 国产一区二区三区视频在线观看| 97精品国产aⅴ7777| 久久久电影免费观看完整版| 欧美精品免费在线| 日韩在线观看高清| 成人写真福利网| 97久久伊人激情网| 亚洲精品成人久久电影| 国产福利精品av综合导导航| 国产日产久久高清欧美一区| 日韩一级黄色av| 亚洲国产成人精品久久| 在线观看国产精品淫| 国产精品自拍偷拍视频| 久久久久久久久久久av| 在线视频欧美性高潮| 亚洲欧美在线第一页| 中文字幕在线视频日韩| 精品一区二区三区电影| 中文字幕日韩欧美| 欧美日韩在线视频观看| 国产一区二区视频在线观看| 国产精品海角社区在线观看| 亚洲第一网中文字幕| 日韩成人激情在线| 九九精品视频在线| 伊人成人开心激情综合网| 97国产在线观看| 日韩69视频在线观看| 日韩小视频在线观看| 亚洲黄色在线观看| 夜夜嗨av一区二区三区四区| 亚洲国产三级网| 国产一区二区三区网站| 日韩精品欧美国产精品忘忧草| 国产在线视频一区| 亚洲欧美在线免费| 久久国产加勒比精品无码| 亚洲天堂av在线免费观看| 尤物tv国产一区| 国产精品入口尤物| 亚洲色图25p| 日韩av免费网站| 在线中文字幕日韩| www国产亚洲精品久久网站| 亚洲欧美日韩国产成人| 亚洲第一视频网站| 欧美视频免费在线观看| 丰满岳妇乱一区二区三区| 欧美电影在线免费观看网站| 操91在线视频| 国产日韩视频在线观看| 亚洲高清色综合| 一区二区三区回区在观看免费视频| 亚洲欧美制服第一页| 中文字幕av日韩| 国产亚洲精品久久久久久777| 97视频免费在线看| 欧美性猛交xxxx免费看久久久| 国产精品无码专区在线观看| 国产精品美女呻吟| 国产香蕉精品视频一区二区三区| 日韩久久精品电影| 欧美性xxxxhd| 日韩av中文字幕在线播放| 欧美成人在线免费视频| 精品美女永久免费视频| 久久精品人人做人人爽| 中文字幕视频一区二区在线有码| 黑人巨大精品欧美一区免费视频| 91av国产在线| 成人性生交xxxxx网站| 久久精品亚洲94久久精品| 久久视频免费观看| 91av视频在线观看| 欧美人与物videos| 狠狠操狠狠色综合网| 亚洲伊人久久综合| 日韩中文综合网| 九色91av视频| 日韩毛片在线看| 欧美日韩国产综合视频在线观看中文| 中文字幕在线视频日韩| 亚洲欧美制服第一页| 久久亚洲精品小早川怜子66| 色综合久久久888| 91久久精品国产| 日本精品久久中文字幕佐佐木| 欧美网站在线观看| 最新91在线视频| 国产精品专区h在线观看| 91精品久久久久久久久久| 欧美日韩福利视频| 欧美人交a欧美精品| 久久97精品久久久久久久不卡| 国产成人拍精品视频午夜网站| 亚洲第一男人av| 国产精品99久久久久久白浆小说| 日韩在线视频观看正片免费网站| 国产精品入口夜色视频大尺度| 亚洲欧美精品中文字幕在线| 欧美中文字幕精品| 夜夜躁日日躁狠狠久久88av| 日韩免费av一区二区| 欧美丰满少妇xxxxx做受| 韩国一区二区电影| 欧美日韩精品在线播放|