C++虛析構(gòu)函數(shù)的使用分析
更新時(shí)間:2013年05月29日 10:29:58 作者:
本篇文章是對(duì)C++虛析構(gòu)函數(shù)的使用進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
在C++中,不能聲明虛構(gòu)造函數(shù),但可以聲明虛析構(gòu)函數(shù)。多態(tài)性是指不同的對(duì)象對(duì)同一消息有不同的行為特性。虛函數(shù)作為運(yùn)行時(shí)多態(tài)性的基礎(chǔ),主要是針對(duì)對(duì)象的,而構(gòu)造函數(shù)是在對(duì)象產(chǎn)生之前運(yùn)行的,因此虛構(gòu)造函數(shù)是沒(méi)有意義的。
析構(gòu)函數(shù)的功能是在該類對(duì)象消亡之前進(jìn)行一些必要的清理工作,析構(gòu)函數(shù)最好都是virtual的。
首先解釋一下虛構(gòu)函數(shù)和指針之間是如何交互的,以及虛析構(gòu)函數(shù)的具體含義。例如以下代碼,其中SomeClass是含有非virtual析構(gòu)函數(shù)的一個(gè)類:
SomeClass *p= new SomeClass;
. . . . . .
delect p;
為p調(diào)用delect時(shí),會(huì)自動(dòng)調(diào)用SomeClass類的析構(gòu)函數(shù)?,F(xiàn)在,再來(lái)看看將析構(gòu)函數(shù)標(biāo)記為virtual之后,會(huì)發(fā)生什么事情。
描述析構(gòu)函數(shù)與虛函數(shù)機(jī)制的交互時(shí),最簡(jiǎn)單的表述是:將所有析構(gòu)函數(shù)都視為具有相同的名字(即使它們并非真的同名)。例如,假定Derived類是Base類的一個(gè)派生類,并假定Base類中的析構(gòu)函數(shù)標(biāo)記為virtual?,F(xiàn)在來(lái)分析以下代碼:
Base *pBase= new Derived;
. . . . . .
delect pBase;
為Base調(diào)用delect時(shí),會(huì)調(diào)用一個(gè)析構(gòu)函數(shù)。由于Base類中的析構(gòu)函數(shù)標(biāo)記為virtual,而且指向的對(duì)象屬于Derived類型,所以會(huì)調(diào)用Derived類中的析構(gòu)函數(shù)。
應(yīng)注意的一點(diǎn)是,將析構(gòu)函數(shù)標(biāo)記為virtual后,派生類所有的析構(gòu)函數(shù)都自動(dòng)成為virtual的(不管時(shí)候用virtual來(lái)標(biāo)記它們)。同樣地,這種行為就好象所有析構(gòu)函數(shù)都具有相同的名字(即使事實(shí)上不同名)。
下面說(shuō)的是將所有析構(gòu)函數(shù)都標(biāo)記為virtual的好處。假定Base類有一個(gè)指針類型的成員變量pB,Base類的構(gòu)造函數(shù)會(huì)創(chuàng)建由pB指向的一個(gè)動(dòng)態(tài)變量,而Base類的析構(gòu)函數(shù)會(huì)刪除pB指向的動(dòng)態(tài)變量。另外,假定Base類沒(méi)有標(biāo)記為virtual,并假定Derived類(它從Base派生)有一個(gè)指針類型的成員變量pD,Derived類的構(gòu)造函數(shù)會(huì)創(chuàng)建有pD指向的一個(gè)動(dòng)態(tài)變量,而Derived類的析構(gòu)函數(shù)會(huì)刪除pD指向的動(dòng)態(tài)變量。分析一下以下代碼:
Base *pBase= new Derived;
. . . . . .
delect pBase;
由于基類中的析構(gòu)函數(shù)沒(méi)有標(biāo)記為virtual,所以只會(huì)調(diào)用Base類的析構(gòu)函數(shù)。它會(huì)將pB指向的動(dòng)態(tài)變量占用的內(nèi)存返還給自由存儲(chǔ)。但是,對(duì)于pD指向的動(dòng)態(tài)變量來(lái)說(shuō),它占用的內(nèi)存永遠(yuǎn)不會(huì)返還給自由存儲(chǔ)(除非程序終止)。
另一方面,如果基類Base的析構(gòu)函數(shù)標(biāo)記為virtual,那么為pBase調(diào)用delect時(shí),就會(huì)調(diào)用Derived類的析構(gòu)函數(shù)(因?yàn)橹赶虻膶?duì)象屬于Derived類型)。Derived類的析構(gòu)函數(shù)會(huì)刪除pD指向的動(dòng)態(tài)變量,再自動(dòng)調(diào)用基類Base的析構(gòu)函數(shù)。后者會(huì)刪除pB指向的動(dòng)態(tài)變量。因此,將基類析構(gòu)函數(shù)標(biāo)記為virtual之后,所有內(nèi)存都能成功地由自由存儲(chǔ)回收。為了準(zhǔn)備好迎接這種情況,最好總是將析構(gòu)函數(shù)標(biāo)記為virtual。
舉個(gè)例子說(shuō)明一下:
#include "stdafx.h"
#include <iostream>
using namespace std;
class Base
{
public:
Base(){cout << " Constructor in Base. " << endl;}
virtual ~Base(){ cout << " Destructor in Base. " << endl;}
};
class Derived:public Base
{
public:
Derived(){cout << " Constructor in Derived. " << endl;}
~Derived(){cout << "Destructor in Derived. " << endl;}
};
int _tmain(int argc, _TCHAR* argv[])
{
Base *p = new Derived;
delete p;
return 0;
}
輸出:
Constructor in Base.
Constructor in Derived.
Destroctor in Derived.
Destroctor in Base.
如果Base中的析構(gòu)函數(shù),沒(méi)有virtual修飾,輸出為:
Constructor in Base.
Constructor in Derived.
Destroctor in Base.
這樣子類Derived中的析構(gòu)函數(shù)沒(méi)有執(zhí)行,會(huì)造成內(nèi)存泄露,因此,如果一個(gè)類是其他類的基類,應(yīng)該將其析構(gòu)函數(shù)聲明為虛析構(gòu)函數(shù)。另外從本例中也可以看出,構(gòu)造函數(shù)、析構(gòu)函數(shù)的執(zhí)行順序。構(gòu)造函數(shù),先基類后子類,析構(gòu)函數(shù),先子類,后基類。
析構(gòu)函數(shù)的功能是在該類對(duì)象消亡之前進(jìn)行一些必要的清理工作,析構(gòu)函數(shù)最好都是virtual的。
首先解釋一下虛構(gòu)函數(shù)和指針之間是如何交互的,以及虛析構(gòu)函數(shù)的具體含義。例如以下代碼,其中SomeClass是含有非virtual析構(gòu)函數(shù)的一個(gè)類:
SomeClass *p= new SomeClass;
. . . . . .
delect p;
為p調(diào)用delect時(shí),會(huì)自動(dòng)調(diào)用SomeClass類的析構(gòu)函數(shù)?,F(xiàn)在,再來(lái)看看將析構(gòu)函數(shù)標(biāo)記為virtual之后,會(huì)發(fā)生什么事情。
描述析構(gòu)函數(shù)與虛函數(shù)機(jī)制的交互時(shí),最簡(jiǎn)單的表述是:將所有析構(gòu)函數(shù)都視為具有相同的名字(即使它們并非真的同名)。例如,假定Derived類是Base類的一個(gè)派生類,并假定Base類中的析構(gòu)函數(shù)標(biāo)記為virtual?,F(xiàn)在來(lái)分析以下代碼:
Base *pBase= new Derived;
. . . . . .
delect pBase;
為Base調(diào)用delect時(shí),會(huì)調(diào)用一個(gè)析構(gòu)函數(shù)。由于Base類中的析構(gòu)函數(shù)標(biāo)記為virtual,而且指向的對(duì)象屬于Derived類型,所以會(huì)調(diào)用Derived類中的析構(gòu)函數(shù)。
應(yīng)注意的一點(diǎn)是,將析構(gòu)函數(shù)標(biāo)記為virtual后,派生類所有的析構(gòu)函數(shù)都自動(dòng)成為virtual的(不管時(shí)候用virtual來(lái)標(biāo)記它們)。同樣地,這種行為就好象所有析構(gòu)函數(shù)都具有相同的名字(即使事實(shí)上不同名)。
下面說(shuō)的是將所有析構(gòu)函數(shù)都標(biāo)記為virtual的好處。假定Base類有一個(gè)指針類型的成員變量pB,Base類的構(gòu)造函數(shù)會(huì)創(chuàng)建由pB指向的一個(gè)動(dòng)態(tài)變量,而Base類的析構(gòu)函數(shù)會(huì)刪除pB指向的動(dòng)態(tài)變量。另外,假定Base類沒(méi)有標(biāo)記為virtual,并假定Derived類(它從Base派生)有一個(gè)指針類型的成員變量pD,Derived類的構(gòu)造函數(shù)會(huì)創(chuàng)建有pD指向的一個(gè)動(dòng)態(tài)變量,而Derived類的析構(gòu)函數(shù)會(huì)刪除pD指向的動(dòng)態(tài)變量。分析一下以下代碼:
Base *pBase= new Derived;
. . . . . .
delect pBase;
由于基類中的析構(gòu)函數(shù)沒(méi)有標(biāo)記為virtual,所以只會(huì)調(diào)用Base類的析構(gòu)函數(shù)。它會(huì)將pB指向的動(dòng)態(tài)變量占用的內(nèi)存返還給自由存儲(chǔ)。但是,對(duì)于pD指向的動(dòng)態(tài)變量來(lái)說(shuō),它占用的內(nèi)存永遠(yuǎn)不會(huì)返還給自由存儲(chǔ)(除非程序終止)。
另一方面,如果基類Base的析構(gòu)函數(shù)標(biāo)記為virtual,那么為pBase調(diào)用delect時(shí),就會(huì)調(diào)用Derived類的析構(gòu)函數(shù)(因?yàn)橹赶虻膶?duì)象屬于Derived類型)。Derived類的析構(gòu)函數(shù)會(huì)刪除pD指向的動(dòng)態(tài)變量,再自動(dòng)調(diào)用基類Base的析構(gòu)函數(shù)。后者會(huì)刪除pB指向的動(dòng)態(tài)變量。因此,將基類析構(gòu)函數(shù)標(biāo)記為virtual之后,所有內(nèi)存都能成功地由自由存儲(chǔ)回收。為了準(zhǔn)備好迎接這種情況,最好總是將析構(gòu)函數(shù)標(biāo)記為virtual。
舉個(gè)例子說(shuō)明一下:
復(fù)制代碼 代碼如下:
#include "stdafx.h"
#include <iostream>
using namespace std;
class Base
{
public:
Base(){cout << " Constructor in Base. " << endl;}
virtual ~Base(){ cout << " Destructor in Base. " << endl;}
};
class Derived:public Base
{
public:
Derived(){cout << " Constructor in Derived. " << endl;}
~Derived(){cout << "Destructor in Derived. " << endl;}
};
int _tmain(int argc, _TCHAR* argv[])
{
Base *p = new Derived;
delete p;
return 0;
}
輸出:
Constructor in Base.
Constructor in Derived.
Destroctor in Derived.
Destroctor in Base.
如果Base中的析構(gòu)函數(shù),沒(méi)有virtual修飾,輸出為:
Constructor in Base.
Constructor in Derived.
Destroctor in Base.
這樣子類Derived中的析構(gòu)函數(shù)沒(méi)有執(zhí)行,會(huì)造成內(nèi)存泄露,因此,如果一個(gè)類是其他類的基類,應(yīng)該將其析構(gòu)函數(shù)聲明為虛析構(gòu)函數(shù)。另外從本例中也可以看出,構(gòu)造函數(shù)、析構(gòu)函數(shù)的執(zhí)行順序。構(gòu)造函數(shù),先基類后子類,析構(gòu)函數(shù),先子類,后基類。
相關(guān)文章
QT實(shí)戰(zhàn)之打開最近文檔功能的實(shí)現(xiàn)
這篇文章主要為大家詳細(xì)介紹了如何利用Qt實(shí)現(xiàn)打開最近文檔功能,并實(shí)現(xiàn)基本的新建、打開、保存、退出、幫助等功能,感興趣的可以動(dòng)手嘗試一下2022-06-06
C++使用boost::lexical_cast進(jìn)行數(shù)值轉(zhuǎn)換
這篇文章介紹了C++使用boost::lexical_cast進(jìn)行數(shù)值轉(zhuǎn)換的方法,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-06-06
詳解如何在VS2019和VScode中配置C++調(diào)用python接口
這篇文章主要介紹了詳解如何在VS2019和VScode中配置C++調(diào)用python接口,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12
Qt實(shí)現(xiàn)部件透明陰影效果與不規(guī)則窗體詳解
這篇文章主要為大家詳細(xì)介紹了Qt實(shí)現(xiàn)部件透明陰影效果與不規(guī)則窗體的相關(guān)方法,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2023-01-01
C語(yǔ)言鏈表實(shí)現(xiàn)歌手評(píng)分系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言鏈表實(shí)現(xiàn)歌手評(píng)分系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-03-03
C++中的多態(tài)問(wèn)題—理解虛函數(shù)表及多態(tài)實(shí)現(xiàn)原理
這篇文章主要介紹了C++中的多態(tài)問(wèn)題—理解虛函數(shù)表及多態(tài)實(shí)現(xiàn)原理,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-02-02
一篇文章帶你了解C語(yǔ)言文件操作中的幾個(gè)函數(shù)
這篇文章主要介紹了使用C語(yǔ)言操作文件的基本函數(shù)整理,包括創(chuàng)建和打開以及關(guān)閉文件的操作方法,需要的朋友可以參考下,希望能夠給你帶來(lái)幫助2021-09-09

