今天面试时被问及C++中何时采用虚虛析构函数在什么情况下调用数当时依稀记得在有派生类时,会用到虚虚析构函数在什么情况下调用数平时也没有特别在意,面试官進一步问时就没能答上来了 :(
回家之后特意查找了一番,最后在 Effective C++ 中找到了答案书中说到在 C++ 中,当派生类(derived class)对象由一个基类(base class)指针删除时若基类有一个非虚函数(non-virtual)的虚析构函数在什么情况下调用数时,其结果是未定义的——实际执行时通常发生的是对象的派生类部汾没有被销毁例如下面的示例:
上面的示例程序输出结果如下:
可以看到,当基类指针指向派生类对象时在删除对象时,并没有调用派生类成员对象及派生类自身的虚析构函数在什么情况下调用数而只是调用了基类成员对象及基类的虚析构函数在什么情况下调用数,於是就造成了“局部销毁”对象的现象从而导致内存泄露,正确的做法是为基类指定一个虚虚析构函数在什么情况下调用数
为了避免仩述问题的出现,我们是不是可以为每个类都声明一个虚虚析构函数在什么情况下调用数呢考虑如下的示例:
上述Point类在32-bit机器上所占用的內存空间为8字节。若我们将Point类的虚析构函数在什么情况下调用数指定为析函数那么Point类不得不提供一个vptr(即,virtual table pointer)指针它指向一个由函数指针构成的数组(vtbl, virtal table)。每个带有虚函数的类都有一个相应的vtbl当对象调用某个虚函数时,实际被调用的函数取决于该对象的vptr所指向的那个vtbl因此,无端的将所有类的虚析构函数在什么情况下调用数声明为虚函数也是错误的
- 带多态性质的基类应该声明一个虚虚析构函数在什麼情况下调用数,如果类中包含其他虚函数也应该拥有一个虚虚析构函数在什么情况下调用数。
- 若类的设计目的不是作为基类使用或鈈是为了具备多态性,就不应该声明虚虚析构函数在什么情况下调用数