C++:14---多级混合继承、虚基类详解

一、多级混合继承

下面介绍菱形继承

//菱形继承
class A
{
public:
    int data;
};
class B:public A
{
public:
    int data;
};
class C:public A
{
public:
    int data;
};
class D:public B,public C
{
public:
    int data;
};

int main()
{
    D c;
    D.data=1;
    D.B::data=2;//访问B中的
    D.C.::data=3;//访问C中的
    D.B::A::data=4;//访问B继承的A
    D.C::A::data=5;//访问C继承的A

    D.A::data=4;//错误,产生二义性,不知道是B、C中哪一个
}
  •  类的内存大小
sizeof(D); //20
sizeof(B); //8
sizeof(C); //8
  • 内存图解 

D先继承于B再继承于C,所以B的数据放在D内存段的最前方,C放在B的后面,D放在最后。

此种菱形继承多存储了两倍的A的内存段,下面将介绍虚基类

C++:14---多级混合继承、虚基类详解_第1张图片

二、虚基类(virtual)

1.概念:也称虚继承、菱形继承。用于多级混合继承时,保留一个虚基类

2.构造顺序

  • 先构造虚基类,如果有多个虚基类,按声明(从左至右)依次构造
  • 再构造基类,如果有多个基类,按声明(从左至右)依次构造
  • 如果有子对象,再构造子对像,如果有多个子对象,按声明的顺序(从上至下)依次构造
  • 最后构造自己
class A //虚基类
{
public:
    A(int data){}//1
    int data;
};
class B:virtual public A
{
public:
    B():A(1){}//2
    int data;
};
class C:virtual public A
{
public:
    C():A(2){}//3
    int data;
};
class D:public B,public C
{
public:
    D():A(3){}//4
    int data;
};
int main()
{
    D d;
    d.data=1;
    d.B::data=2;
    d.C::data=3;

    d.B::A.data=4;
    d.C::A::data=5;
}
  • 构造顺序为:1-2-3-4
  • 构造顺序解释:构造类D对象d的时候,发现继承于B,于是去构造B,构造B的时候,发现继承于虚基类A,于是构造虚基类A,接着构造B。再接着构造C,发现C继承于虚基类A,但发现虚基类A已经被B构造过了,所以不再构造A,直接构造C。最后构造D
  • 如果虚基类构造函数为带参构造,则其子类,以及子类拓展出来的子类,都要在成员初始化列表对其进行构造函数的初始化
  • d的data赋值为1,继承于B、C,分别赋值为2,3,顺序为从左至右。A的值本来为4,后来执行到最后一行的时候被赋值为5

3.类的内存大小

  • 继承于虚基类的类,内存地址大小加4字节(此4字节是属于虚基类的,不是属于自己的)
sizeof(B); //12
sizeof(C); //12
sizeof(D); //24

4.内存地址图解

  • 虚基类的内存地址在派生类内存地址的最后
  • 虚基类在派生时,只保存一份内存在派生类内存中

C++:14---多级混合继承、虚基类详解_第2张图片

 地址解析:

  • B和C中都保存了A的值,但是在D继承B和C的时候,只保存了一份A,且放在最后
  • 在D继承的B和C内存段中分别有一个函数指针放在最前方

你可能感兴趣的:(C++)