当前位置导航:炫浪网>>网络学院>>编程开发>>C++教程>>C++进阶与实例

C++对象布局及多态实现探索之虚继承

  下面我们来看虚继承。首先看看这C020类,它从C010虚继承: 

 struct C010
{
 C010() : c_(0x01) {}
 void foo() { c_ = 0x02; }
 char c_;
};
struct C020 : public virtual C010
{
 C020() : c_(0x02) {}
 char c_;
};

  运行如下代码,查看对象的内存布局:
 
 PRINT_SIZE_DETAIL(C020)

  结果为:
 
 The size of C020 is 6
The detail of C020 is c0 c2 45 00 02 01

  很明显对象的起始处是一个指针,然后是子类的成员变量,接下来是父类的成员变量。和以前的讨论不同的是由于使用了虚继承,父类的成员变量被放到了最后面。
 
  运行如下的代码:
 
 C020 c020;
c020.C010::c_ = 0x04;

  由于子类中的变量和父类中的变量重名,所以我们必须用这种方式来访问属于父类的成员变量,普通情况下不需要这种写法。我们看看后面这行代码对应的汇编代码:
 
 0042387E mov eax,dword ptr [ebp+FFFFF82Ch]
00423884 mov ecx,dword ptr [eax+4]
00423887 mov byte ptr [ebp+ecx+FFFFF82Ch],4


  前面说过对象的起始是一个指针,第1行指令取到这个指针的值,第2行把这个指针指向的地址后移4字节后的值(做为一个4字节的值)取出来。执行完这句我们看看ecx寄存器,可知取出来的值为5.最后一行是真正的赋值指令,它通过在对象的起始处(即[ebp+FFFFF32Ch])加上ecx中的值做偏移值(即5)来得到赋值的目的地址。接合前面的对象布局输出,我们可以发现从对象起始地址开始加5字节的偏移值,刚好得到父类的成员变量的地址。这样我们可以大致分析出直接虚继承的子类的对象布局。
 

共2页 首页 上一页 1 2 下一页 尾页 跳转到
相关内容
赞助商链接