這個(gè)問(wèn)題在實(shí)踐中偶爾會(huì)碰到,設(shè)計(jì)一個(gè)TimeKeeper基類和一些派生類來(lái)記錄時(shí)間:
1 class TimeKeeper
2 {
3 public:
4 TimeKeeper();
5 ~TimeKeeper();
6
7 };
8
9 class AtomicClock: public TimeKeeper {}; //原子鐘
10 class WaterClock: public TimeKeeper {}; //水鐘
在使用時(shí),我們可能會(huì)使用factory工廠方法:
1 TimeKeeper* getTimeKeeper();//返回一個(gè)指針,指向一個(gè)派生類的動(dòng)態(tài)分配的對(duì)象
2
3 TimeKeeper* ptk = getTimeKeeper();//從繼承體系中得到一個(gè)動(dòng)態(tài)分配對(duì)象
4
5 delete ptk;//負(fù)責(zé)的刪除它
刪除的時(shí)候就會(huì)出現(xiàn)問(wèn)題,因?yàn)閜tk這個(gè)指針指向的是基類,那刪除的指令會(huì)執(zhí)行基類TimeKeeper的析構(gòu)函數(shù),該函數(shù)不是virtual函數(shù)。
在C++中,這樣的情況下其刪除行為沒(méi)有被定義,一般會(huì)只刪除基類的成分,而派生類的那些元素沒(méi)有被刪除,加入收藏這就是形成資源泄露,敗壞數(shù)據(jù)結(jié)構(gòu),在調(diào)試器上浪費(fèi)很多時(shí)間的絕佳途徑哦(引用原文翻譯)。
解決的方法就是定義一個(gè)基類的virtual析構(gòu)函數(shù),這樣一來(lái),刪除行為就會(huì)在派生類中實(shí)現(xiàn),不會(huì)只刪除一部分。
一般來(lái)說(shuō),只要類中有virtual函數(shù),就要定義一個(gè)virtual析構(gòu)函數(shù)。不過(guò),如果類中沒(méi)有virtual函數(shù),就不需要也不應(yīng)該定義virtual析構(gòu)函數(shù),這樣不僅沒(méi)用,而且也會(huì)增加額外開(kāi)支,且會(huì)產(chǎn)生很多的兼容性問(wèn)題,因?yàn)関irtual機(jī)制是c++特有的。
另外,C++中很多類的實(shí)現(xiàn)都是不帶virtual的,比如:string,STL中的vector,list,set,trl::unordered_map,如果繼承它們很可能出現(xiàn)上述的錯(cuò)誤,所以提醒大家:拒絕繼承標(biāo)準(zhǔn)容器或者其它只有非virtual析構(gòu)函數(shù)的類!
相關(guān)推薦:
2009年4月計(jì)算機(jī)等級(jí)二級(jí)考試VF程序設(shè)計(jì)輔導(dǎo) C語(yǔ)言輔導(dǎo)三種常見(jiàn)的中文內(nèi)碼的轉(zhuǎn)換方法 計(jì)算機(jī)等考二級(jí)C語(yǔ)言考前復(fù)習(xí)資料(for循環(huán))