#include <iostream>
using namespace std;
class Base
{
public:
~Base()
{
cout<<"~Base"<<endl;
}
};
class Derived :public Base
{
public:
Derived()
{
m_char = new char[10];
}
~Derived()
{
delete[]m_char;
cout<<"~Derived"<<endl;
}
private:
char* m_char;
};
int main()
{
Base *pBase = new Derived;
delete pBase;
return 0;
}
如以上的代码,运行结果为:
~Base
1、如果将Base的析构函数申明为虚函数,即:virtual ~Base(){}
执行结果为:
~Derived
~Base
2、总结:
1、delete指向派生类对象的基类指针时,如基类析构非虚,则不会调用派生类的析构函数,而只调用基类的析构函数。
2、基类的析构函数申明为虚函数时,delete指向派生类指针的基类指针时,会调用派生类的析构函数。
3、如上面的例子,如Base的析构函数非虚,那么,将造成m_char内存泄漏。
4、如果基类的析构函数为虚函数,那么它的派生类析构函数会自动变为虚函数(即使不显式加virtual关键字,不过最好加上)。
5、虚析构函数,可以理解为名字“不同”的普通虚函数,因为编译器可以自动辨识类的析构函数,我们可以将派生类的析构函数理解为基类析构函数不同名的普通虚函数。
6、虚函数,是通过virtual table来维护的,所以将析构函数申明为虚函数,则必然要产生虚函数表,会造成额外内存开销,所以在实际的应用过程中,如果我们没有通过基类指针释放派生类对象的情况,尽量不要将基类的析构申明为虚函数。