新聞中心
C++中虛析構(gòu)函數(shù)的作用是什么,針對(duì)這個(gè)問(wèn)題,這篇文章詳細(xì)介紹了相對(duì)應(yīng)的分析和解答,希望可以幫助更多想解決這個(gè)問(wèn)題的小伙伴找到更簡(jiǎn)單易行的方法。
成都創(chuàng)新互聯(lián)公司專(zhuān)注為客戶(hù)提供全方位的互聯(lián)網(wǎng)綜合服務(wù),包含不限于成都網(wǎng)站設(shè)計(jì)、網(wǎng)站制作、石泉網(wǎng)絡(luò)推廣、成都小程序開(kāi)發(fā)、石泉網(wǎng)絡(luò)營(yíng)銷(xiāo)、石泉企業(yè)策劃、石泉品牌公關(guān)、搜索引擎seo、人物專(zhuān)訪、企業(yè)宣傳片、企業(yè)代運(yùn)營(yíng)等,從售前售中售后,我們都將竭誠(chéng)為您服務(wù),您的肯定,是我們最大的嘉獎(jiǎng);成都創(chuàng)新互聯(lián)公司為所有大學(xué)生創(chuàng)業(yè)者提供石泉建站搭建服務(wù),24小時(shí)服務(wù)熱線:028-86922220,官方網(wǎng)址:www.cdcxhl.com
一.虛析構(gòu)函數(shù)的作用
總的來(lái)說(shuō)虛析構(gòu)函數(shù)是為了避免內(nèi)存泄露,而且是當(dāng)子類(lèi)中會(huì)有指針成員變量時(shí)才會(huì)使用得到的。也就說(shuō)虛析構(gòu)函數(shù)使得在刪除指向子類(lèi)對(duì)象的基類(lèi)指針時(shí)可以調(diào)用子類(lèi)的析構(gòu)函數(shù)達(dá)到釋放子類(lèi)中堆內(nèi)存的目的,而防止內(nèi)存泄露的.
我們知道,用C++開(kāi)發(fā)的時(shí)候,用來(lái)做基類(lèi)的類(lèi)的析構(gòu)函數(shù)一般都是虛函數(shù)。可是,為什么要這樣做呢?下面用一個(gè)小例子來(lái)說(shuō)明:
#includeusing namespace std; class ClxBase { public: ClxBase() {}; virtual ~ClxBase() { cout<<"delete ClxBase"< DoSomething(); delete pTest; return 0; }
但是,如果把類(lèi)ClxBase析構(gòu)函數(shù)前的virtual去掉,那輸出結(jié)果就是下面的樣子了:
沒(méi)有調(diào)動(dòng)子類(lèi)的析構(gòu)函數(shù)
也就是說(shuō),類(lèi)ClxDerived的析構(gòu)函數(shù)根本沒(méi)有被調(diào)用!一般情況下類(lèi)的析構(gòu)函數(shù)里面都是釋放內(nèi)存資源,而析構(gòu)函數(shù)不被調(diào)用的話(huà)就會(huì)造成內(nèi)存泄漏。我想所有的C++程序員都知道這樣的危險(xiǎn)性。當(dāng)然,如果在析構(gòu)函數(shù)中做了其他工作的話(huà),那你的所有努力也都是白費(fèi)力氣。
所以,文章開(kāi)頭的那個(gè)問(wèn)題的答案就是--這樣做是為了當(dāng)用一個(gè)基類(lèi)的指針刪除一個(gè)派生類(lèi)的對(duì)象時(shí),派生類(lèi)的析構(gòu)函數(shù)會(huì)被調(diào)用。
當(dāng)然,并不是要把所有類(lèi)的析構(gòu)函數(shù)都寫(xiě)成虛函數(shù)。因?yàn)楫?dāng)類(lèi)里面有虛函數(shù)的時(shí)候,編譯器會(huì)給類(lèi)添加一個(gè)虛函數(shù)表,里面來(lái)存放虛函數(shù)指針,這樣就會(huì)增加類(lèi)的存儲(chǔ)空間。所以,只有當(dāng)一個(gè)類(lèi)被用來(lái)作為基類(lèi)的時(shí)候,才把析構(gòu)函數(shù)寫(xiě)成虛函數(shù)。
總結(jié)一下虛析構(gòu)函數(shù)的作用:
(1)如果父類(lèi)的析構(gòu)函數(shù)不加virtual關(guān)鍵字
當(dāng)父類(lèi)的析構(gòu)函數(shù)不聲明成虛析構(gòu)函數(shù)的時(shí)候,當(dāng)子類(lèi)繼承父類(lèi),父類(lèi)的指針指向子類(lèi)時(shí),delete掉父類(lèi)的指針,只調(diào)動(dòng)父類(lèi)的析構(gòu)函數(shù),而不調(diào)動(dòng)子類(lèi)的析構(gòu)函數(shù)。
(2)如果父類(lèi)的析構(gòu)函數(shù)加virtual關(guān)鍵字
當(dāng)父類(lèi)的析構(gòu)函數(shù)聲明成虛析構(gòu)函數(shù)的時(shí)候,當(dāng)子類(lèi)繼承父類(lèi),父類(lèi)的指針指向子類(lèi)時(shí),delete掉父類(lèi)的指針,先調(diào)動(dòng)子類(lèi)的析構(gòu)函數(shù),再調(diào)動(dòng)父類(lèi)的析構(gòu)函數(shù)。
二.虛析構(gòu)函數(shù)的原理分析
#includeusing namespace std; class Base { public: Base(){cout<<"create Base"< 從創(chuàng)建講起,用gdb調(diào)試你會(huì)發(fā)現(xiàn),
(1)先調(diào)用父類(lèi)的構(gòu)造函數(shù),再調(diào)用子類(lèi)的構(gòu)造函數(shù),
這里有一個(gè)問(wèn)題:父類(lèi)的構(gòu)造函數(shù)/析構(gòu)函數(shù)與子類(lèi)的構(gòu)造函數(shù)/析構(gòu)函數(shù)會(huì)形成多態(tài),但是當(dāng)父類(lèi)的構(gòu)造函數(shù)/析構(gòu)函數(shù)即使被聲明virtual,子類(lèi)的構(gòu)造/析構(gòu)方法仍無(wú)法覆蓋父類(lèi)的構(gòu)造方法和析構(gòu)方法。這是由于父類(lèi)的構(gòu)造函數(shù)和析構(gòu)函數(shù)是子類(lèi)無(wú)法繼承的,也就是說(shuō)每一個(gè)類(lèi)都有自己獨(dú)有的構(gòu)造函數(shù)和析構(gòu)函數(shù)。
(2)而由于父類(lèi)的析構(gòu)函數(shù)為虛函數(shù),所以子類(lèi)會(huì)在所有屬性的前面形成虛表,而虛表內(nèi)部存儲(chǔ)的就是父類(lèi)的虛函數(shù)
即使子類(lèi)也有虛函數(shù),但是由于是單繼承,所以也只有一張?zhí)摫恚@在上一篇博客多態(tài)中講到過(guò)。
執(zhí)行 Base *b = new Der;之后b的最終形態(tài)
(3)當(dāng)delete父類(lèi)的指針時(shí),由于子類(lèi)的析構(gòu)函數(shù)與父類(lèi)的析構(gòu)函數(shù)構(gòu)成多態(tài),所以得先調(diào)動(dòng)子類(lèi)的析構(gòu)函數(shù);之所以再調(diào)動(dòng)父類(lèi)的析構(gòu)函數(shù),是因?yàn)閐elete的機(jī)制所引起的,delete 父類(lèi)指針?biāo)傅目臻g,要調(diào)用父類(lèi)的析構(gòu)函數(shù)。
所以結(jié)果就是這樣
關(guān)于C++中虛析構(gòu)函數(shù)的作用是什么問(wèn)題的解答就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,如果你還有很多疑惑沒(méi)有解開(kāi),可以關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道了解更多相關(guān)知識(shí)。
分享名稱(chēng):C++中虛析構(gòu)函數(shù)的作用是什么
文章來(lái)源:http://www.ef60e0e.cn/article/ghpiij.html