[原創(chuàng)文章歡迎轉(zhuǎn)載,但請(qǐng)保留作者信息]
Justin 于 2009-11-02
vczh同學(xué)建議發(fā)布隨筆的時(shí)候選擇首頁(yè),不知合不合適,我先試試看,要是有不對(duì)的地方,還請(qǐng)指點(diǎn):)
從Item7開(kāi)始講析構(gòu)函數(shù)(Destructor)。
將析構(gòu)函數(shù)定義為虛函數(shù)(virtual function)是必須要提的。(是不是需要一個(gè)虛函數(shù)的筆記?)
在一個(gè)多態(tài)(Polymorphic)的繼承體系中,基類(lèi)(base class)的析構(gòu)函數(shù)應(yīng)該定義為虛函數(shù)。否則在通過(guò)基類(lèi)指針析構(gòu)一個(gè)子類(lèi)對(duì)象的時(shí)候,因?yàn)闆](méi)有虛表(V Table)對(duì)析構(gòu)函數(shù)的指引,對(duì)象的基類(lèi)部分是被毀了,對(duì)象的子類(lèi)部分卻沒(méi)辦法得到析構(gòu)(沒(méi)有了虛表中的說(shuō)明它根本不知道有子類(lèi)的析構(gòu)函數(shù)來(lái)析構(gòu)子類(lèi)部分!),從而導(dǎo)致內(nèi)存泄漏。
我想炒冷飯系列應(yīng)該以舉例為特色……好吧,我就姑且來(lái)胡亂舉個(gè)例子:
假設(shè)基類(lèi)指針在析構(gòu)時(shí)要做的事情是奉命救人:指針啊指針,XX地發(fā)大水了,你趕緊去把張三一家人(子類(lèi)對(duì)象)救出來(lái)。
指針拿著指令(析構(gòu)函數(shù))就找到了張三(張三家家長(zhǎng),子類(lèi)對(duì)象的基類(lèi)部分):張三是吧,你一家人都在這里了吧,跟我走吧。(析構(gòu)吧,回空余內(nèi)存里去吧@#¥%)
如果張三已經(jīng)沒(méi)有其他親戚了,他就是張三家的全部人(這個(gè)對(duì)象其實(shí)就是基類(lèi)對(duì)象)。很簡(jiǎn)單,直接跟著指針走就完事了。
如果張三還有個(gè)小弟叫張三瘋(張三家除張三外的一部分,即子類(lèi)對(duì)象的子類(lèi)部分),張三就必須拿著族譜(V Table)跟指針說(shuō)哎呀這是我弟啊,把他也析構(gòu)了吧,張三家對(duì)您感恩不盡……也很簡(jiǎn)單,一并帶走就完成任務(wù)了(完整析構(gòu)/釋放了整個(gè)對(duì)象)
如果,張三沒(méi)了族譜,記不得有個(gè)張三瘋弟弟,傻乎乎的跟了指針走,大家也以為張三一家都被救出來(lái)了。災(zāi)區(qū)里就只剩下張三瘋一個(gè)人在瘋跑了(memory leak)
另一方面,如果不準(zhǔn)備把某個(gè)類(lèi)用作多態(tài)中的基類(lèi),就沒(méi)必要定義析構(gòu)函數(shù)為虛函數(shù)。這樣除了增加代碼大小開(kāi)銷(xiāo)外(因?yàn)樾枰臻g存放虛表),沒(méi)有任何好處。(這個(gè)倒是很多書(shū)上沒(méi)說(shuō)過(guò)的)
如果張三家本來(lái)就只有張三一個(gè)人,他就沒(méi)必要還要帶著個(gè)空白的族譜了。逃命的時(shí)候能扔就扔嘛。
最后提到的一點(diǎn)和抽象類(lèi)有關(guān)。抽象類(lèi)是有純虛函數(shù)的類(lèi),只能作為接口,不能有自己的對(duì)象。在設(shè)計(jì)軟件框架的時(shí)候會(huì)有需要用到抽象類(lèi)的時(shí)候。可是如果有這么一個(gè)類(lèi),需要設(shè)計(jì)為抽象類(lèi),但是卻找不到一個(gè)成員函數(shù)可以拉去做純虛函數(shù),那么這個(gè)時(shí)候就把析構(gòu)函數(shù)定義為純虛函數(shù)好了。只是同時(shí)還要給它一個(gè)空的實(shí)現(xiàn)。
class::~class() {}
這種情況應(yīng)該還是有的,只是我現(xiàn)在卻想象不出來(lái)……那就先死記著咯~