基類和派生類指針轉(zhuǎn)換--dynamic_cast
1、背景
編程中遇到如下問(wèn)題:有基類指針pA指向派生類B對(duì)象,使用該指針調(diào)用虛函數(shù),執(zhí)行的是派生類中的函數(shù),沒(méi)問(wèn)題。不過(guò)現(xiàn)在想執(zhí)行派生類中非虛函數(shù),因?yàn)樵撝羔樖腔愵愋偷?,所以?zhí)行的是基類中函數(shù)或者是返回錯(cuò)誤(基類中沒(méi)有此函數(shù))。當(dāng)然,在這種確定此基類指針指向的是派生類對(duì)象的情況下,可以將此基類指針強(qiáng)制類型轉(zhuǎn)換為派生類指針:B *pB=(B *)pA。
在查找相關(guān)資料時(shí),學(xué)到了基類和派生類指針轉(zhuǎn)換方法dynamic_cast:如果基類指針確實(shí)是指向了一個(gè)派生類對(duì)象,此運(yùn)算符會(huì)傳回轉(zhuǎn)換后的派生類指針,否則,返回空指針?!径嗔说罊z查,比強(qiáng)制類型轉(zhuǎn)換安全】不過(guò)要使用dynamic_cast,需要編譯器允許時(shí)間類型信息(RTTI)。
2、網(wǎng)上轉(zhuǎn)的一點(diǎn)資料
C++程序員大多喜歡使用強(qiáng)制類型轉(zhuǎn)換(我也是),盡管它是C遺留下來(lái)的,盡管它存在這樣那樣的缺點(diǎn),但是你不能不承認(rèn)它使用起來(lái)很方便,而且絕大多數(shù)情況下是不會(huì)產(chǎn)生問(wèn)題的.極少數(shù)情況下可能會(huì)存在類型轉(zhuǎn)換失敗的情況,這時(shí)候就需要使用到dynamic_cast了,這里提到的"極少數(shù)情況"是這樣的:如果有繼承或多重繼承的類對(duì)象,你在某些情況下得到某個(gè)對(duì)象的指針,而你又想將其轉(zhuǎn)換為某個(gè)特定類型,但是由于C++中對(duì)象類型的多態(tài)性(它可以是多種類型),你又不能確定(在運(yùn)行時(shí))這么做一定會(huì)成功,此時(shí)可以使用dynamic_cast,充分利用C++的運(yùn)行時(shí)檢查機(jī)制.只是用語(yǔ)言描述太抽象了,舉個(gè)例子吧.
class?A{...};? class?B:public?A{...};? class?C:public?B{...};? void?Fun1(B*?pB)? {? A*?pA??=?(A*)pB;? C*?pC??=?(C*)pB;? ...? }? ??Fun1函數(shù)使用強(qiáng)制類型轉(zhuǎn)換將pB轉(zhuǎn)換為A*或C*,看出什么的問(wèn)題了嗎? ??如果這樣調(diào)用Fun1: ?????????????????????????????Fun1(((B*)new C)); 的確不會(huì)有問(wèn)題,但如果是這樣呢:? ?????????????????????????????Fun1(new B); pC不會(huì)為NULL,能夠想到使用pC指針時(shí)就程序就悲劇了. 更嚴(yán)重情況下,如果是這樣:? ?????????????????????????????Fun1((B*)0X00005678);//0X00005678是一個(gè)隨機(jī)值 pA,PC就不會(huì)是NULL,強(qiáng)制類型轉(zhuǎn)換總是能夠成功的,但使用這兩個(gè)指針時(shí)程序肯定崩潰.當(dāng)然你可以使用異常處理機(jī)制來(lái)處理這樣的錯(cuò)誤,不過(guò)這有點(diǎn)大才小用的感覺(jué),最好能夠找到一種能夠檢查出類型轉(zhuǎn)換能否成功的辦法.這時(shí)dynamic_cast就能大顯身手了.
A*?pA??=?dynamic_cast