【c++】深度探索虚继承内存布局



C++中虚拟继承的概念


为了解决从不同途径继承来的同名的数据成员在内存中有不同的拷贝造成数据不一致问题,将共同基类设置为虚基类。这时从不同的路径继承过来的同名数据成员在内存中就只有一个拷贝,同一个函数名也只有一个映射。这样不仅就解决了二义性问题,也节省了内存,避免了数据不一致的问题。
class 派生类名:virtual 继承方式  基类名
virtual是关键字,声明该基类为派生类的虚基类。
在多继承情况下,虚基类关键字的作用范围和继承方式关键字相同,只对紧跟其后的基类起作用。
声明了虚基类之后,虚基类在进一步派生过程中始终和派生类一起,维护同一个基类子对象的拷贝。


C++虚拟继承

◇概念:

C++使用虚拟继承(Virtual Inheritance),解决从不同途径继承来的同名的数据成员在内存中有不同的拷贝造成数据不一致问题,将共同基类设置为虚基类。这时从不同的路径继承过来的同名数据成员在内存中就只有一个拷贝,同一个函数名也只有一个映射。

 

◇解决问题:

解决了二义性问题,也节省了内存,避免了数据不一致的问题。
 
  
 
  
虚基类(把一个动词当成一个名词而已)
当在多条继承路径上有一个公共的基类,在这些路径中的某几条汇合处,这个公共的基类就会产生多个实例(或多个副本),若只想保存这个基类的一个实例,可以将这个公共基类说明为虚基类。

 

【c++】深度探索虚继承内存布局_第1张图片


【c++】深度探索虚继承内存布局_第2张图片

derivedClass类对象中存在两份baseClass成员变量,这显然不是我们需要的。会产生二义性、同时也会浪费内存资源。


如何解决这种问题呢,使用虚拟继承,

#include
#include
#include
using namespace std;

class A {
public:
    A(){printf("A create.\n");}
    int a;
    virtual void fun(){}
};

class B:virtual public A{
public:
    B(){printf("B create.\n");}
    int b;
    virtual void fun1(){}
};

class C:virtual public A
{
public :
    int c;
    C(){printf("C create.\n");}
    virtual void fun3(){printf("fun3 .\n");}
};

class D:public C,public B{
public:
    int d;
    D(){printf("D create.\n");}
    virtual void fun3(){printf("fun4 .\n");}
}; 
//二义性问题的开销 
int main() {
    D *pd=new D;
    pd->B::a=1;
    pd->C::a=2;
    printf("%d\n",pd->B::a);
    printf("%d\n",pd->C::a);
    printf("%d\n",sizeof(D));
    getchar();
}


【c++】深度探索虚继承内存布局_第3张图片

在这个对象模型中,有个virtual base class数组指针,里面内容如下图:

【c++】深度探索虚继承内存布局_第4张图片

相信看完这个图,你对菱形虚拟继承内存对象模型能有更深的理解。



你可能感兴趣的:(【深入C++】)