C++ 析构函数设为虚函数的好处
[轉] 一個帖子,很能說明這個問題
問題:?
以下這段代碼執(zhí)行后會不會有問題??
基類:?
class CBase??
{?
public:?
??? virtual void VirtualFun1(CString strFun1) = 0;?
};?
子類:?
class CDerived : public CBase??
{?
public:?
??? CDerived();?
??? virtual ~CDerived();????
??? virtual void VirtualFun1(CString strFun1);?
private:?
??? CString m_strFun1;?
};?
CDerived::CDerived()?
{?
}?
CDerived::~CDerived()?
{?
}?
void CDerived::VirtualFun1(CString strFun1)?
{?
??? m_strFun1 = strFun1;?
}?
客戶代碼:?
??? CBase *pBase = new CDerived();?
??? pBase->VirtualFun1("test");?
??? delete pBase;?
答案/分析:?
會!?
只要一運行,就會發(fā)生內存泄漏(我的測試平臺:VC6);?
調試器輸出信息:?
Dumping objects ->?
strcore.cpp(118) : {76} normal block at 0x003A36E8, 16 bytes long.?
Data: <??????????? tsr > 01 00 00 00 03 00 00 00 03 00 00 00 74 73 72 00?
Object dump complete.?
注:?
01 00 00 00 03 00 00 00 03 00 00 00 74 73 72 00?
這部分數(shù)據(jù)可能是調試器的內存保護數(shù)據(jù),未驗證?
問題表面是在于m_strFun1,問題的實質其實是CBase 沒有提供虛擬析構函數(shù),只要為CBase 加入一個虛擬析構函數(shù)即解決了上面的問題;?
原因就在于當delete pBase的時候,只是調用了CBase的析構函數(shù)(非虛擬,編譯器背地里完成,對于純虛擬類還沒有驗證過),從而導致未調用CDerived的析構函數(shù),進而造成m_strFun1沒有被釋放內存。?
事后,查了一下資料,其實在《Effective C++》中的條款14中對此進行了詳細的解釋,有興趣的可以去參考一下。?
如果單單分析這段代碼,其實還是好解決的,如果把這段代碼放到一個大程序中,出現(xiàn)類似的內存泄漏時,可能就很難追查了,所以還是要打好基礎,寫好每一行代碼。?
總結:?
1. C++基本功不扎實,對析構函數(shù)了解不夠,對編譯器的內部行為不夠了解;?
2. 忽視了代碼的變化,特別是向導生成的代碼,一般通過向導會自動建立一個虛擬析構函數(shù);?
3. 發(fā)現(xiàn)問題的關鍵比解決問題更難;由于這個內存泄漏是發(fā)生在一個比較大的程序中,并且還涉及到其它dll,花了很長時間對它進行定位,確定發(fā)生泄漏的代碼行;?
4. 調試是對基本知識的綜合考驗,最能反映一個人的技術深度,深入剝析后就能加深技術細節(jié)的理解和記憶;
總結
以上是生活随笔為你收集整理的C++ 析构函数设为虚函数的好处的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 基于嗅探原理的原始套接字木马
- 下一篇: C++中空指针调用类成员函数的原理