c++類型轉(zhuǎn)換及RTTI運(yùn)行階段類型識(shí)別
正文
我們都知道C++完全兼容C語(yǔ)言,C語(yǔ)言的轉(zhuǎn)換方式很簡(jiǎn)單,可以在任意類型之間轉(zhuǎn)換,但這也恰恰是缺點(diǎn),因?yàn)闃O其不安全,可能不經(jīng)意間將指向const對(duì)象的指針轉(zhuǎn)換成非const對(duì)象的指針,可能將基類對(duì)象指針轉(zhuǎn)成了派生類對(duì)象的指針,這種轉(zhuǎn)換很容易出bug,需要嚴(yán)格審查代碼才能消除這種隱患,但是C這種轉(zhuǎn)換方式不利于我們審查代碼,且程序運(yùn)行時(shí)也可能會(huì)出bug。
所以C++引入的這幾種類型轉(zhuǎn)換可以完美的解決上述問(wèn)題,不同場(chǎng)景下不同需求使用不同的類型轉(zhuǎn)換方式,同時(shí)有利于代碼審查。
孫悟空都只有七十二變,不能瞎變,所以c++給類型轉(zhuǎn)換做了限制。
1、static_cast
static_cast
僅當(dāng)type_name可以被隱式轉(zhuǎn)換為expression所屬類型或expression可隱式轉(zhuǎn)換成type_name所屬類型時(shí),上述轉(zhuǎn)換才是合法的。
static_cast是用得最多的一類類型轉(zhuǎn)換符,常見(jiàn)的枚舉值轉(zhuǎn)成整形,float轉(zhuǎn)整形之類的,都是可以的。
另外,static_cast還可以將派生類指針轉(zhuǎn)換為基類指針,而且一定條件下還能將基類指針轉(zhuǎn)換為派生類指針,且不會(huì)報(bào)錯(cuò),只是一些只有派生類才會(huì)有的函數(shù)、成員變量,轉(zhuǎn)換過(guò)來(lái)的指針也不會(huì)有
Test test;
TestDerived derived;
cout << "----------" << endl;
Test* tp = static_cast<Test*>(&derived);
tp->func();
cout << "-----------------" << endl;
TestDerived* dp = static_cast<TestDerived*>(&test);
dp->func();
dp->speak();
//以下是控制臺(tái)輸出
----------
TestDerived func
-----------------
test func2、dynamic_cast
dynamic_cast運(yùn)算符的語(yǔ)法和static_cast一樣,但它的作用和static_cast略有區(qū)別。
kotlin中有個(gè)語(yǔ)法叫 is,本人覺(jué)得dynamic_cast就是kotlin中的is。
dynamic_cast,一般只用于基類和派生類之間的轉(zhuǎn)換,而且只能用于派生類指針轉(zhuǎn)換成基類指針,不能反向轉(zhuǎn)換
if (Test* tpp = dynamic_cast<Test*>(&derived)) {
cout << "devived can cast to test" << endl;
}
if (TestDerived* dpp = dynamic_cast<TestDerived*>(&test)) {
cout << "test can cast to TestDerived" << endl;
}
//輸出
devived can cast to test如代碼所示,如果dynamic_cast轉(zhuǎn)換成功,將返回一個(gè)指針,如果轉(zhuǎn)換失敗,將返回一個(gè)空指針。所以代碼中的兩個(gè)if判斷,只有一個(gè)生效??催@種調(diào)用方式,是不是和kotlin中的 is 很相象呢
3、const_cast
const_cast運(yùn)算符,只用于執(zhí)行一種用途的類型轉(zhuǎn)換,即改變值為const或volatile。
它一般用于去const運(yùn)算符。但去運(yùn)算符之后的效果卻是難以預(yù)料。
const int num = 10;
const int* tempN = const_cast<const int*>(&num);
cout << "tempn = " << *tempN << endl;
int* temppp = const_cast<int*>(tempN);
*temppp = 20;
cout << "num = " << num << " tempn = " << *tempN << " temppp = " << *temppp << endl;
輸出:
tempn = 10
num = 10 tempn = 20 temppp = 20如上述代碼所示,num是const類型的整形值,它的值始終為10,無(wú)法更改。這種轉(zhuǎn)換慎用
4、reinterpret_cast
沒(méi)有啥特殊場(chǎng)景運(yùn)用,類似于c語(yǔ)言中的強(qiáng)制轉(zhuǎn)換,一般用得極少。
5、RTTI
RTTI,運(yùn)行階段類型識(shí)別的簡(jiǎn)稱。
在多態(tài)中,比如上面代碼中有基類Test和TestDerived,現(xiàn)在有一個(gè)Test指針,但不知道這個(gè)指針究竟指向的是基類還是派生類,怎么知道指針是指向的哪種對(duì)象呢?
這就是RTTI的工作,在運(yùn)行時(shí)判斷類型。目前c++中有3個(gè)支持RTTI的元素:
- dynamic_cast,將一個(gè)指向基類的指針來(lái)生成一個(gè)指向派生類的指針,否則,該運(yùn)算符將返回空指針
- typeid,返回一個(gè)指針對(duì)象類型的值
- type_info,結(jié)構(gòu)存儲(chǔ)了有關(guān)特定類型的信息
RTTI場(chǎng)景下,父類必須要有虛函數(shù)信息,因?yàn)镽TTI信息存儲(chǔ)在虛函數(shù)表中。
以上就是c++類型轉(zhuǎn)換及RTTI運(yùn)行階段類型識(shí)別的詳細(xì)內(nèi)容,更多關(guān)于c++類型轉(zhuǎn)換RTTI的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
C++簡(jiǎn)明圖解分析靜態(tài)成員與單例設(shè)計(jì)模式
與靜態(tài)數(shù)據(jù)成員不同,靜態(tài)成員函數(shù)的作用不是為了對(duì)象之間的溝通,而是為了能處理靜態(tài)數(shù)據(jù)成員,靜態(tài)成員函數(shù)沒(méi)有this指針。既然它沒(méi)有指向某一對(duì)象,也就無(wú)法對(duì)一個(gè)對(duì)象中的非靜態(tài)成員進(jìn)行默認(rèn)訪問(wèn)2022-06-06
詳解Matlab實(shí)現(xiàn)動(dòng)態(tài)表白圖的繪制
這篇文章主要利用Matlab實(shí)現(xiàn)繪制獨(dú)特的表白動(dòng)圖,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Matlab有一定的幫助,感興趣的小伙伴可以了解一下2022-05-05
C++11中value category(值類別)及move semantics(移動(dòng)語(yǔ)義)的介紹
這篇文章主要給大家介紹了C++11中value category(值類別)及move semantics(移動(dòng)語(yǔ)義)的介紹,文中介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2018-05-05
C/C++編譯報(bào)錯(cuò)printf was not declared in 
這篇文章主要介紹了C/C++編譯報(bào)錯(cuò)printf was not declared in this scope問(wèn)題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08
C++?AVL樹(shù)插入新節(jié)點(diǎn)后的四種調(diào)整情況梳理介紹
AVL樹(shù)是高度平衡的而二叉樹(shù),它的特點(diǎn)是AVL樹(shù)中任何節(jié)點(diǎn)的兩個(gè)子樹(shù)的高度最大差別為1,本文主要給大家介紹了C++如何實(shí)現(xiàn)AVL樹(shù),需要的朋友可以參考下2022-08-08
C++常用函數(shù)之XML JSON格式轉(zhuǎn)換問(wèn)題
XML在Json出現(xiàn)前應(yīng)用很廣泛,靈活性好,應(yīng)用語(yǔ)言也沒(méi)有限制,發(fā)展了這么長(zhǎng)時(shí)間后xml標(biāo)準(zhǔn)已經(jīng)很臃腫。這篇文章主要介紹了C++常用函數(shù)之XML JSON格式轉(zhuǎn)換問(wèn)題,需要的朋友可以參考下2020-02-02

