在<<反C++>>文中提到的 VC6.0 上的编译器支持一个d1reportAllClassLayout的开关, 能够输出全部对象的内存布局信息, 我自己经常使用vs2005来开发,所以这里我就以vs2005为例讲怎么设置d1reportAllClassLayout。
右键项目属性(Properties)—》配制属性(Configuration Properties)—》C/C++--》命令行(Command Line)的框里输入/d1reportAllClassLayout,就可以看类的对象布局。例如以下图:(当然也能够使用命令行:cl –d1reportSingleClassLayout[classname] test.cpp)
測试代码:
class CBase
{
public:
virtual void fun(void) {}
private:
int m_valuable;
};
输出结果:
class CBase size(8):
+---
0 | {vfptr}
4 | m_valuable
+---
CBase::$vftable@:
| &CBase_meta
| 0
0 | &CBase::fun
CBase::fun this adjustor: 0
CDerived:非virtual继承CBase,持有一个virtual重写方法和一个virtual新方法。一个成员变量
class CDerived: public CBase
{
public:
void fun(void) { }
virtual void vfun(void) { }
public:
int m_derived;
};
编译后输出结果:
class CDerived size(12):
+---
| +--- (base class CBase)
0 | | {vfptr}
4 | | m_valuable
| +---
8 | m_derived
+---
CDerived::$vftable@:
| &CDerived_meta
| 0
0 | &CDerived::fun
1 | &CDerived::vfun
CDerived::fun this adjustor: 0
CDerived::vfun this adjustor: 0
当中vftable@CBase的CBase::fun项在这里更新为CDerived::fun,同一时候添加了一项CDerived::vfun。
CDerived2:virtual继承CBase,持有一个virtual重写方法和一个virtual新方法,一个成员变量。因此结构为vftable@自身,vbtable@自身,member@自身,以及CBase结构。