虚基类
class B
{
public:
int b;
};
class C1 :virtual public B
{
public:
int c1;
};
class C2 :virtual public B
{
public:
int c2;
};
class D :public C1, public C2
{
public:
int d;
};
int main()
{
cout << sizeof(D ) << endl; //24
B b;
D d;
system( "pause");
return 0;
}
在类D中添加函数,改为:
class D :public C1, public C2
{
public:
void Display()
{
c1 = 0x02;
C2::b = 0x03;
c2 = 0x04;
d = 0x05;
cout << "this=" << this << endl << endl;
cout << "&C1::b=" << &(C1 ::b) << endl;
cout << "&c1=" << &c1 << endl;
cout << "&C2::b=" << &(C2 ::b) << endl;
cout << "&c2=" << &c2 << endl;
cout << "&d=" << &d << endl;
cout << b << endl;
}
int d;
};
主函数写为:
int main()
{
cout << sizeof(D ) << endl; //24
B b;
D d;
d.Display();
C1 c;
system( "pause");
return 0;
}
调试截图为:
原理分析:
因为类B是类C1和类C2的基类,所以C1和C2都会继承B的数据成员b,所以C1和C2中会存在同名成员,当类D继承类C1和类C2的时候,就会有两个数据成员b,这样会保留多份数据成员的拷贝,不仅占用较多的存储空间,还增加了访问这些成员的困难,实际上,我们也不需要有多份拷贝。
为了解决这一问题,我们让类C1和类C2在继承B的时候进行了虚继承,即在继承方式前加virtual ,虚基类使得在继承间接共同基类时只保留一份成员。
数据结果分析:
在代码中,我们只给类C1中继承的b赋值为3,但是截图显示,类C1和类C2中的数据成员b的值都为3,而且类C1和类C2 中数据成员Bde地址是一样的,这就验证了"虚基类使得在继承间接共同基类时只保留一份成员 "
虚基类并不是在声明基类时声明的,而是在声明派生类时,指定继承方式时声明的。
声明虚基类的一般方式为:
class 派生类名:virtual 继承方式 基类名
在最后的派生类中,不仅要负责对其直接基类进行初始化,还要负责对虚基类初始化
对于这个例子
有虚基类的时候:
没有虚基类的时候:
没有虚基类的时候,对于这种菱形继承,派生类D中会有两个数据成员int b,也是因此,在访问的时候必须要写清楚访问的是那个类里面的b成员,如C1::b,否则会因为访问成员不明确而造成错误,为了避免这一问题,要在声明继承方式的时候加上virtual关键字,表示这是一个虚继承
通过看内存,可以清楚的看到在类D中,最后继承下来的数据成员是如何放置的
c1中的虚基类地址存储的偏移量20
c2中的虚基类地址存储的偏移量12
在内存中大概就是这个样子: