一 。虚基类表内容继续分析
class Grand{
public:
int c;
}
class A1 : virtual public Grand{ //虚继承
public:
}
class A2 : virtual public Grand{ //虚继承
public:
}
class C1 :public A1, public A2{
public:
}
int main(){
cout << sizeof(Grand) <<endl; // 4
cout << sizeof(A1) <<endl; // 8
cout << sizeof(A2) <<endl; // 8
cout << sizeof(C1) <<endl; // 12
}
/*
虚基类表 一般是八字节 四字节为一个单位
没多一个虚基类 就会增加4字节
编译器因为有虚基类 会给A1 A2 增加默认的构造函数 并且这个默认构造函数里 会被编译器增加进去代码
给vbptr(虚基类表指针) 赋值
A1 myclassA1;
myclassA1.c = 12;
这个赋值过程和之前的虚函数赋值过程一样都是this指针加上偏移量进行赋值
不过需要注意的是grand中的c 在内存布局的最下面
*/
-----------------------------------------
class Grand{
public:
int c1;
}
class Grand2{
public:
int c2;
}
class A1 : virtual public Grand,public Grand2{ //虚继承
public:
int c3;
}
class A2 : virtual public Grand { //虚继承
public:
}
class C1 :public A1, public A2{
public:
}
int main(){
cout << sizeof(Grand) <<endl; // 4
cout << sizeof(A1) <<endl; // 16 int c2 vbptr int c3 int c1
}
// 不管虚继承在前还是实继承在前 虚继承的内存布局永远都是在最后面
实继承的数据会在内存布局中处于上游
自身数据数据会在内存布局中处于中游
虚继承的数据会在内存布局中处于下游
同样如果都是虚继承则 排练顺序为 虚基类表指针 int c3 int c1 int c2
只有对虚基类成员进行处理 比如赋值的时候才会用到虚基类表取其中的偏移 参与地址运算