Item 7: Declare destructors virtual in polymorphic base classes
析构函数声明为虚函数恐怕是面试中最常见的问题之一。目的在于以基类指针调用析构函数时能够正确地析构子类部分的内存。 否则子类部分的内存将会泄漏,正确的用法如下:
// 基类
class TimeKeeper{
public:
virtual ~TimeKeeper();
...
};
TimeKeeper *ptk = getTimeKeeper(): // 可能返回任何一种子类
...
delete ptk;
值得一提的是虚函数表指针的内存占用问题,我们知道所有存在虚方法的对象中都会存有一个虚函数表指针vptr
, 用来在运行时定位虚函数。同时,每个存在虚方法的类也会对应一个虚函数列表的指针vtbl
。 函数调用时会在vtbl
指向的虚函数表中寻找vptr
指向的那个函数。
为了准确地描述vptr
的大小,先来了解一下对象的sizeof
计算方式,以及字节对齐问题:
编译器:gcc version 5.1.0,目标平台:x86_64-apple-darwin14.3.0
没有任何成员的对象也需要有地址,否则怎么定位它呢?所以编译器会给它一字节的大小:
class C0{};
sizeof(C0); // 1
class C1{
char i;
};
sizeof(C1); // 1
只有一个char
成员的对象大小也是1
整个对象的大小会按照最大的成员进行字节对齐。例如:
class C2{
char i, j;
};
sizeof(C2); // 2
class C3{
char i, j;
int k;
};
sizeof(C3); // 8
因为int
大小是4,两个char
大小是2,故总的大小以4为基对齐,大小为4*2 = 8。
class C4{
char i;
virtual void func();
};
sizeof(C4); // 16
因为我的Target是64位平台,故vptr
的大小为8,char
大小为1,故总的大小以8为基对齐,大小为8*2 = 16。 虚函数指针不仅使得对象更加占用内存空间,同时会造成可移植性问题。 问题很明显:一个包含vptr
的C++对象传递给Fortran时,由于Fortran中没有vptr
的概念, 因此需要重新计算对象大小,然而vptr
的大小是平台相关的。。。
包含对象成员的类称为封闭类,封闭类以对象成员中最大的基本数据类型的长度进行字节对齐。例如:
class C5{
C4 c4;
char i;
};
sizeof(C5); // 24
C4和C5中最大的基本数据类型是void*
(vptr
的类型),其大小为8,故以8为基对齐的结果是8*3 = 24。
除非注明,本博客文章均为原创,转载请以链接形式标明本文地址: http://harttle.com/2015/07/24/effective-cpp-7.html