C++類(lèi)型轉(zhuǎn)換總結(jié)
C++類(lèi)型轉(zhuǎn)換總結(jié)
C風(fēng)格的強(qiáng)制類(lèi)型轉(zhuǎn)換(Type Cast)很簡(jiǎn)單,不管什么類(lèi)型的轉(zhuǎn)換統(tǒng)統(tǒng)是:
TYPE b = (TYPE)a。
C++風(fēng)格的類(lèi)型轉(zhuǎn)換提供了4種類(lèi)型轉(zhuǎn)換操作符來(lái)應(yīng)對(duì)不同場(chǎng)合的應(yīng)用。
const_cast,字面上理解就是去const屬性。
static_cast,命名上理解是靜態(tài)類(lèi)型轉(zhuǎn)換。如int轉(zhuǎn)換成char。
dynamic_cast,命名上理解是動(dòng)態(tài)類(lèi)型轉(zhuǎn)換。如子類(lèi)和父類(lèi)之間的多態(tài)類(lèi)型轉(zhuǎn)換。
reinterpret_cast,僅僅重新解釋類(lèi)型,但沒(méi)有進(jìn)行二進(jìn)制的轉(zhuǎn)換。
4種類(lèi)型轉(zhuǎn)換的格式,如:TYPE B = static_cast(TYPE)(a)。
const_cast
去掉類(lèi)型的const或volatile屬性。
?
1?struct?SA?{ 2?????int?i; 3?}; 4?const?SA?ra; 5?//ra.i?=?10;?//直接修改const類(lèi)型,編譯錯(cuò)誤 6?SA?&rb?=?const_cast(ra); 7?rb.i?=?10;
?
static_cast
類(lèi)似于C風(fēng)格的強(qiáng)制轉(zhuǎn)換。無(wú)條件轉(zhuǎn)換,靜態(tài)類(lèi)型轉(zhuǎn)換。用于:
1. 基類(lèi)和子類(lèi)之間轉(zhuǎn)換:其中子類(lèi)指針轉(zhuǎn)換成父類(lèi)指針是安全的;但父類(lèi)指針轉(zhuǎn)換成子類(lèi)指針是不安全的。(基類(lèi)和子類(lèi)之間的動(dòng)態(tài)類(lèi)型轉(zhuǎn)換建議用dynamic_cast)
2. 基本數(shù)據(jù)類(lèi)型轉(zhuǎn)換。enum,struct, int, char, float等。static_cast不能進(jìn)行無(wú)關(guān)類(lèi)型(如非基類(lèi)和子類(lèi))指針之間的轉(zhuǎn)換。
3. 把空指針轉(zhuǎn)換成目標(biāo)類(lèi)型的空指針。
4. 把任何類(lèi)型的表達(dá)式轉(zhuǎn)換成void類(lèi)型。
5. static_cast不能去掉類(lèi)型的const、volitale屬性(用const_cast)。
1?int?n?=?6; 2?double?d?=?static_cast(n);?//?基本類(lèi)型轉(zhuǎn)換 3?int?*pn?=?&n; 4?double?*d?=?static_cast(&n)?//無(wú)關(guān)類(lèi)型指針轉(zhuǎn)換,編譯錯(cuò)誤 5?void?*p?=?static_cast(pn);?//任意類(lèi)型轉(zhuǎn)換成void類(lèi)型
dynamic_cast
有條件轉(zhuǎn)換,動(dòng)態(tài)類(lèi)型轉(zhuǎn)換,運(yùn)行時(shí)類(lèi)型安全檢查(轉(zhuǎn)換失敗返回NULL):
1. 安全的基類(lèi)和子類(lèi)之間轉(zhuǎn)換。
2. 必須要有虛函數(shù)。
3. 相同基類(lèi)不同子類(lèi)之間的交叉轉(zhuǎn)換。但結(jié)果是NULL。
?
?1?class?BaseClass?{ ?2?public: ?3?int?m_iNum; ?4?virtual?void?foo(){};?//基類(lèi)必須有虛函數(shù)。保持多臺(tái)特性才能使用dynamic_cast ?5?}; ?6? ?7?class?DerivedClass:?public?BaseClass?{ ?8?public: ?9?char?*m_szName[100]; 10?void?bar(){}; 11?}; 12? 13?BaseClass*?pb?=?new?DerivedClass(); 14?DerivedClass?*pd1?=?static_cast(pb);?//子類(lèi)->父類(lèi),靜態(tài)類(lèi)型轉(zhuǎn)換,正確但不推薦 15?DerivedClass?*pd2?=?dynamic_cast(pb);?//子類(lèi)->父類(lèi),動(dòng)態(tài)類(lèi)型轉(zhuǎn)換,正確 16? 17?BaseClass*?pb2?=?new?BaseClass(); 18?DerivedClass?*pd21?=?static_cast(pb2);?//父類(lèi)->子類(lèi),靜態(tài)類(lèi)型轉(zhuǎn)換,危險(xiǎn)!訪(fǎng)問(wèn)子類(lèi)m_szName成員越界 19?DerivedClass?*pd22?=?dynamic_cast(pb2);?//父類(lèi)->子類(lèi),動(dòng)態(tài)類(lèi)型轉(zhuǎn)換,安全的。結(jié)果是NULL
?
reinterpret_cast
僅僅重新解釋類(lèi)型,但沒(méi)有進(jìn)行二進(jìn)制的轉(zhuǎn)換:
1. 轉(zhuǎn)換的類(lèi)型必須是一個(gè)指針、引用、算術(shù)類(lèi)型、函數(shù)指針或者成員指針。
2. 在比特位級(jí)別上進(jìn)行轉(zhuǎn)換。它可以把一個(gè)指針轉(zhuǎn)換成一個(gè)整數(shù),也可以把一個(gè)整數(shù)轉(zhuǎn)換成一個(gè)指針(先把一個(gè)指針轉(zhuǎn)換成一個(gè)整數(shù),在把該整數(shù)轉(zhuǎn)換成原類(lèi)型的指針,還可以得到原先的指針值)。但不能將非32bit的實(shí)例轉(zhuǎn)成指針。
3. 最普通的用途就是在函數(shù)指針類(lèi)型之間進(jìn)行轉(zhuǎn)換。
4. 很難保證移植性。
1?int?doSomething(){return?0;}; 2?typedef?void(*FuncPtr)();?//FuncPtr?is?一個(gè)指向函數(shù)的指針,該函數(shù)沒(méi)有參數(shù),返回值類(lèi)型為?void 3?FuncPtr?funcPtrArray[10];?//10個(gè)FuncPtrs指針的數(shù)組讓我們假設(shè)你希望(因?yàn)槟承┠涿畹脑颍┌岩粋€(gè)指向下面函數(shù)的指針存入funcPtrArray數(shù)組: 4? 5?funcPtrArray[0]?=?&doSomething;//?編譯錯(cuò)誤!類(lèi)型不匹配,reinterpret_cast可以讓編譯器以你的方法去看待它們:funcPtrArray 6?funcPtrArray[0]?=?reinterpret_cast(&doSomething);?//不同函數(shù)指針類(lèi)型之間進(jìn)行轉(zhuǎn)換
總結(jié)
去const屬性用const_cast。
基本類(lèi)型轉(zhuǎn)換用static_cast。
多態(tài)類(lèi)之間的類(lèi)型轉(zhuǎn)換用daynamic_cast。
不同類(lèi)型的指針類(lèi)型轉(zhuǎn)換用reinterpret_cast。