《effective C++》中的條款14:確定基類有虛析構函數。也就是說,如果某個類要作為基類來使用,一般使用虛析構函數。看下面的例子:
class A{
public:
A(){
strA = new char[10];
strcpy(strA, "abc");
}
virtual ~A(){
cout<<"~A() " <<strA<<endl; delete []strA;
}
private:
char* strA;
};
class B: public A
{
public:
B(){
strB = new char[10];
strcpy(strB, "def");
}
~B(){
cout<<"~B() " <<strB<<endl;
delete []strB;
}
private:
char* strB;
};
測試:
A *p = new B; delete p;
代碼的輸出結果:
~A()
我們發現B類的析構函數沒有調用,導致內存泄露。(因為動態編譯,在運行時會檢查有無派生類對象重載本函數,有則調用之。)
當我們可能通過基類指針刪除派生類對象時,并且被析構的對象是有重要的析構函數的派生類的對象,就需要讓基類的析構函數成為虛擬的。
當一個類不準備作為基類使用時,使析構函數為虛函數一般是個壞主意。因為當類里面有虛函數的時候,編譯器會給類添加一個虛函數表,里面來存放虛函數指針,這樣就會增加類的存儲空間。所以,只有類要作為基類來使用時,才把析構函數寫成虛函數。