c++類型轉換及RTTI運行階段類型識別
正文
我們都知道C++完全兼容C語言,C語言的轉換方式很簡單,可以在任意類型之間轉換,但這也恰恰是缺點,因為極其不安全,可能不經意間將指向const對象的指針轉換成非const對象的指針,可能將基類對象指針轉成瞭派生類對象的指針,這種轉換很容易出bug,需要嚴格審查代碼才能消除這種隱患,但是C這種轉換方式不利於我們審查代碼,且程序運行時也可能會出bug。
所以C++引入的這幾種類型轉換可以完美的解決上述問題,不同場景下不同需求使用不同的類型轉換方式,同時有利於代碼審查。
孫悟空都隻有七十二變,不能瞎變,所以c++給類型轉換做瞭限制。
1、static_cast
static_cast
僅當type_name可以被隱式轉換為expression所屬類型或expression可隱式轉換成type_name所屬類型時,上述轉換才是合法的。
static_cast是用得最多的一類類型轉換符,常見的枚舉值轉成整形,float轉整形之類的,都是可以的。
另外,static_cast還可以將派生類指針轉換為基類指針,而且一定條件下還能將基類指針轉換為派生類指針,且不會報錯,隻是一些隻有派生類才會有的函數、成員變量,轉換過來的指針也不會有
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(); //以下是控制臺輸出 ---------- TestDerived func ----------------- test func
2、dynamic_cast
dynamic_cast運算符的語法和static_cast一樣,但它的作用和static_cast略有區別。
kotlin中有個語法叫 is
,本人覺得dynamic_cast就是kotlin中的is
。
dynamic_cast,一般隻用於基類和派生類之間的轉換,而且隻能用於派生類指針轉換成基類指針,不能反向轉換
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轉換成功,將返回一個指針,如果轉換失敗,將返回一個空指針。所以代碼中的兩個if判斷,隻有一個生效。看這種調用方式,是不是和kotlin中的 is
很相象呢
3、const_cast
const_cast運算符,隻用於執行一種用途的類型轉換,即改變值為const或volatile。
它一般用於去const運算符。但去運算符之後的效果卻是難以預料。
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,無法更改。這種轉換慎用
4、reinterpret_cast
沒有啥特殊場景運用,類似於c語言中的強制轉換,一般用得極少。
5、RTTI
RTTI,運行階段類型識別的簡稱。
在多態中,比如上面代碼中有基類Test和TestDerived,現在有一個Test指針,但不知道這個指針究竟指向的是基類還是派生類,怎麼知道指針是指向的哪種對象呢?
這就是RTTI的工作,在運行時判斷類型。目前c++中有3個支持RTTI的元素:
- dynamic_cast,將一個指向基類的指針來生成一個指向派生類的指針,否則,該運算符將返回空指針
- typeid,返回一個指針對象類型的值
- type_info,結構存儲瞭有關特定類型的信息
RTTI場景下,父類必須要有虛函數信息,因為RTTI信息存儲在虛函數表中。
以上就是c++類型轉換及RTTI運行階段類型識別的詳細內容,更多關於c++類型轉換RTTI的資料請關註WalkonNet其它相關文章!
推薦閱讀:
- C++ 強制類型轉換詳解
- C++ RTTI與4種類型轉換的深入理解
- C++強制類型轉換(static_cast、dynamic_cast、const_cast、reinterpret_cast)
- 淺析C++中dynamic_cast和static_cast實例語法詳解
- C++中4種強制類型轉換的區別詳析