C++类里面,我们经常可以看到析构函数是虚函数,这个虚函数有什么作用吗?我们可以通过一个很简单的例子来看看虚析构函数的作用。
class CBase { public: virtual ~CBase() { printf("CBase::~CBase()\n"); } }; class CChild: public CBase { public: virtual ~CChild() { printf("CChild::~CChild()\n"); } }; int _tmain(int argc, _TCHAR* argv[]) { CChild* p = new CChild(); delete p; CBase* p2 = new CChild(); delete p2; return 0; }
一个基类,一个子类,相当的简单。先看下面2行代码:
CChild* p = new CChild(); delete p;
delete p的时候调用基类还是子类的析构函数呢?其实这个问题很简单,p是CChild指针(子类),那么自然调用子类的析构函数了。基类析构函数会被调用吗?答案是肯定的,其实这个是C++的特性,子类的析构函数会自动调用基类的析构函数。子类析构是这样的过程:
1. 析构子类扩展部分(也就是运行子类析构函数代码);
2. 在子类析构函数返回之前调用基类析构函数来释放基类部分。
有图有真相:
从callstack里面 可以看的很清楚,在delete p的时候,先进入CChild析构函数~CChild(),然后~CChild()又调用了基类析构函数~CBase()。
这个过程跟析构函数是否是虚函数无关。也就是说:
无论析构函数是否是虚函数,子类的析构函数一定会调用基类的析构函数。顺序是先析构子类部分,再析构基类部分。
(对于构造函数,我们可以在子类的构造函数里面选择调用基类的某一个构造函数,如果不在子类里面显式地调用基类构造函数,那么系统自动会调用基类的默认构造函数。顺序刚好和析构相反,先构造基类部分,再构造子类部分)
我们再来看看这几行代码:
CBase* p2 = new CChild(); delete p2;
delete p2的时候调用基类还是子类的析构函数呢?先来看看虚析构函数的情况:
从上面的图中,可以清楚的看到调用的是子类的析构函数。根据我们前面得出来的结论:子类析构函数会自动调用基类析构函数,那么对于这种情况(虚析构函数)就是:
子类和基类的析构函数都会被调用。(其中基类析构函数是被子类析构函数自动调用的)
再来看看非虚析构函数的情况,先看图:
我们会发现子类析构函数~CChild()并没有出现在call stack中,我们只看到了基类析构函数~CBase(),也就是说子类析构函数并没有被调用。
至此我们可以得出结论,对于delete一个指向子类的基类指针:
1. 对于虚析构函数,那么就是基类和子类的析构函数都会被调用,先析构子类部分,再析构基类部分。(基类析构函数是被子类析构函数自动调用的)
2. 对于非虚析构函数,子类析构函数不会被调用,只有基类析构函数才会被调用。
为什么虚析构函数的情况下,子类析构函数会被调用呢?这个地方就涉及到虚函数表的问题。当一个C++类里面有一个或者多个虚函数的时候,C++编译器会给这个类生产一个虚函数表。在这个类的对象里面会自动生成一个成员_vfptr, _vfptr就是指向虚函数表的一个指针。通过这个虚函数表,C++就可以支持多态。至于虚函数表的工作机理,此处就不做探讨了。
通常我们建议把析构函数搞成虚函数,这个是有好处的。不然很多时候会发现子类的析构函数没有被调用,也就是造成了内存泄漏。但是世事无绝对,有时候把析构函数搞成虚函数也会带来一点问题,Effective C++书里面有介绍,大家可以参考。
总之,我们只要搞清楚原理,就可以根据实际情况来取舍了。