C++ RTTI的dynamic_cast函數

這篇文章是C++ RTTI的后續,前面我們介紹了typeid()操作符,這篇文章介紹RTTI的另一個概念,即dynamic_cast。
相比較typeid,dynamic_cast在實際項目中會被大量使用。


首先,一句話dynamic_cast是干什么用的,dynamic_cast是在兩個類之間做類型轉換的,即把一個指針類型轉換成另一個類型;這個轉換過程是在運行時刻轉換的,所以叫dynamic_cast(與此相對的是static_cast)。dynamic_cast的使用必須滿足下面兩個條件:

  1. 如果兩個類之間沒有父子關系,那么source必須是多態的。
  2. 如果是從父類到子類的轉換,那么父類(source)也必須多態的。

否則編譯器就報錯:
<file>:41: error: cannot dynamic_cast 'pc' (of type 'class <SRC> *') to type 'class <DST> *' (source type is not polymorphic)

我們重新解釋一下上述描述:dynamic_cast的使用需要source是多態的,只有一種情況例外,那就是從子類到父類的轉換。再換一個角度,其實dynamic_cast的使用必須要source是多態的,因為同學們想想,例外情況中子類到父類的轉換是天然的行為,根本不需要dynamic_cast啊,像(Parent *)child。
我們看一段代碼:

#include <stdio.h>
#include <string>

class A1 {};
class A2 : public A1 {};

void foo(A2 * pa2) {
    A1 * va1 = dynamic_cast<A1 *>(pa2);
}

int main(int argc, char * argv[]) {
    A2 * a2 = new A2();
    foo(a2);
    return 0;
}

編譯器生成的匯編代碼如下:

_Z3fooP2A2:
    pushq   %rbp
    movq    %rsp, %rbp
    movq    %rdi, -24(%rbp)
    movq    -24(%rbp), %rax
    movq    %rax, -8(%rbp)
    leave
    ret

可見編譯器直接把pa2的賦給了va1,根本沒有調用到dynamic_cast,因為編譯器明確知道父類和子類的關系以及成員構成,又沒有多態的問題,所以編譯器就能夠完成類型轉換工作,其實此時dynamic_cast是被映射成了static_cast使用。


討論完使用場合,下面我們看看轉換的結果(假定source都是多態的):

  1. 如果source和dest沒有父子關系,那么結果是NULL,即使兩個類定義成一模一樣。
  2. 從子類到父類,前面說過必然是成功的。
  3. 從父類到子類,這也是dynamic_cast最常用的情況,這依賴于父類指針是否真實的指向了對應的子類類型,標示這是不是一個真實的子類對象。
#include <stdio.h>
#include <string>

class A {
public:
    virtual ~A() {}
};
class B1: public A {}; 
class B2: public A {}; 

void foo(A * pa) {
    B1 * vb1 = dynamic_cast<B1 *>(pa);
    printf("%s\n", vb1 == NULL ? "NULL" : "NOT NULL");
}

int main(int argc, char * argv[]) {
    B1 * b1 = new B1();
    B2 * b2 = new B2();
    foo(b1);
    foo(b2);
    return 0;
}

運行結果如下:

NOT NULL
NULL

可見b1是能轉換成功的,b2不能成功,因為雖然b1和b2都是A的實例,但是b2不是B1的實例。


前面我們介紹了dynamic_cast的使用場景,現在我們介紹dynamic_cast是如何工作的;代碼說明問題:

#include <stdio.h>
#include <string>

class A {
public:
    virtual ~A() {}
};
class B: public A {};

void foo(A * pa) {
    B * vb = dynamic_cast<B *>(pa);
}

int main(int argc, char * argv[]) {
    B * b = new B();
    foo(b);
    return 0;
}

類A是類B的父類,我們看生成的foo()函數代碼:

_Z3fooP1A:
    pushq   %rbp
    movq    %rsp, %rbp
    subq    $32, %rsp
    movq    %rdi, -24(%rbp)
    movq    -24(%rbp), %rax
    testq   %rax, %rax
    jne .L9
    movl    $0, %eax
    jmp .L10
.L9:
    movl    $0, %ecx
    movl    $_ZTI1B, %edx
    movl    $_ZTI1A, %esi
    movq    %rax, %rdi
    call    __dynamic_cast
.L10:
    movq    %rax, -8(%rbp)
    leave
    ret

首先驗證參數pa是否為NULL,如果是則直接返回NULL,如果不是則調用C++ lib庫函數__dynamic_cast。

庫函數__dynamic_cast需要四個參數:

extern "C" void*
  __dynamic_cast (const void *v,
                  const abi::__class_type_info *src,
                  const abi::__class_type_info *dst,
                  std::ptrdiff_t src2dst_offset)

參數聲明為:

  1. v: source對象地址,NOT NULL(前面源代碼我們看的如果是NULL就不會進到這兒來了),并且由于source是多態的,那么source對象的第一個域是指向虛函數表的指針。
  2. src: source對象的類類型
  3. dat: destination對象的類類型
  4. 這個參數我也沒弄清楚,但是當src是dst的基類時為-2,當src和dst不相干時為0。

關于這個函數的詳細說明搜搜C++ ABI文檔,比如: https://android.googlesource.com/platform/abi/cpp/+/6426040f1be4a844082c9769171ce7f5341a5528/src/dynamic_cast.cc

#define DYNAMIC_CAST_NO_HINT -1
#define DYNAMIC_CAST_NOT_PUBLIC_BASE -2
#define DYNAMIC_CAST_MULTIPLE_PUBLIC_NONVIRTUAL_BASE -3
  /* v: source address to be adjusted; nonnull, and since the
   *    source object is polymorphic, *(void**)v is a virtual pointer.
   * src: static type of the source object.
   * dst: destination type (the "T" in "dynamic_cast<T>(v)").
   * src2dst_offset: a static hint about the location of the
   *    source subobject with respect to the complete object;
   *    special negative values are:
   *       -1: no hint
   *       -2: src is not a public base of dst
   *       -3: src is a multiple public base type but never a
   *           virtual base type
   *    otherwise, the src type is a unique public nonvirtual
   *    base type of dst at offset src2dst_offset from the
   *    origin of dst.
   */

下面我們再看src(class A)和dst(class B)兩個參數的值定義:

_ZTS1B:
    .string "1B"
_ZTS1A:
    .string "1A"
_ZTI1B:
    .quad   _ZTVN10__cxxabiv120__si_class_type_infoE+16
    .quad   _ZTS1B
    .quad   _ZTI1A
_ZTI1A:
    .quad   _ZTVN10__cxxabiv117__class_type_infoE+16
    .quad   _ZTS1A

參數src的類型時$_ZTI1A, 參數dst的類型時$_ZTI1B;從內容可以看出這兩個類型的定義也是不一樣的

  1. $_ZTI1A的類型是__class_type_info
  2. $_ZTI1B的類型是__si_class_type_info,是__class_type_info的子類;子類包含一個指向父類類類型的指針:
    const __class_type_info* __base_type;

參閱 /usr/include/c++/4.4.4/cxxabi.h

所有這些類的類型信息都在應用程序啟動的時候,在main函數入口之前注冊到全局變量里,使得在用戶程序能夠訪問到他們。


最后總結一下dynamic_cast的使用場景

  1. dynamic_cast用來實現指針類型的轉化。
  2. dynamic_cast如果轉化成功則返回指向目標類型的指針,如果轉換不成功返回NULL。
  3. dynamic_cast要求src必須是多態的,因為dynamic_cast需要從類的虛函數表表中獲得類類型信息。
  4. dynamic_cast最常見的用法是從一個抽象基類轉換到具體的實現類。

當一個父類有多種子類時,如果目前有一個指向父類的指針,但是我們不知道指向父類的指針實際上指向的是哪一種子類,可以使用dynamic_cast<Child *>來判斷,如果返回不是NULL,說明這是一個指向Child子類的指針,否則就不是。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,732評論 6 539
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,214評論 3 426
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,781評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,588評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,315評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,699評論 1 327
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,698評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,882評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,441評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,189評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,388評論 1 372
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,933評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,613評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,023評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,310評論 1 293
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,112評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,334評論 2 377

推薦閱讀更多精彩內容

  • C++中的類型轉換分為兩種:隱式類型轉換;顯式類型轉換。 而對于隱式變換,在很多時候,不經意間就發生了,比如int...
    DayDayUpppppp閱讀 14,600評論 6 26
  • C++類型轉換總結 本章內容:1 前言2 static_cast3 dynamic_cast4 const_cas...
    Haley_2013閱讀 955評論 0 50
  • 1. C++基礎知識點 1.1 有符號類型和無符號類型 當我們賦給無符號類型一個超出它表示范圍的值時,結果是初始值...
    Mr希靈閱讀 18,031評論 3 82
  • 一、 C/C++程序基礎 面試例題1——分析代碼寫輸出(一般賦值語句的概念和方法)。 面試例題2—...
    LuckTime閱讀 2,003評論 2 42
  • 山村口,一個年歲稍長的女孩子推著一個木輪椅,輪椅上坐著一個小男孩。 女孩身材修長,姿態優雅,一塊花布方巾包著如瀑的...
    念衡閱讀 313評論 0 4