如何解决多继承下的 菱形继承 问题

简介: 如何解决多继承下的 菱形继承 问题

概念:

8c2aaddf248048c0a6601b5e1df68605.jpg

此时D类属于多继承,可以看到D类里面会有两份A类的数据,菱形继承也并不一定就一定就是上图的菱形,假如B类下面还有一个类,D类继承它,同样也是菱形继承问题

class A
{
public:
  int a;
};
class B : public A
{
public:
  int b;
};
class C : public A
{
public:
  int c;
};
class D : public B, public C
{
public:
  int d;
};
int main()
{
  D data;
  cout << data.a << endl;
  return 0;
}

 大家觉得这段代码有问题嘛?

8b189ba85da940dbb7cdeafc5883362e.jpg

我们此时D类已经出现了数据冗余和二义性的问题了,我们直接访问 a 的值是不可行的,只有明确地表明你要访问地是哪个类里面的 a

class A
{
public:
  int a;
};
class B : virtual public A
{
public:
  int b;
};
class C : virtual public A
{
public:
  int c;
};
class D : public B, public C
{
public:
  int d;
};
int main()
{
  D data;
  cout << data.a << endl;
  return 0;
}

不过可以在腰部,也就是 B和C 类加上 virtual ,就可以数据二义性的问题和数据冗余,这个叫 菱形虚拟继承


菱形虚拟继承:

class A
{
public:
  int a;
};
//class B : virtual public A
class B : public A
{
public:
  int b;
};
//class C : virtual public A
class C : public A
{
public:
  int c;
};
class D : public B, public C
{
public:
  int d;
};
int main()
{
  D data;
  //data.a = 1;
  data.B::a = 2;
  data.C::a = 6;
  data.b = 3;
  data.c = 4;
  data.d = 5;
  return 0;
}

 我们来看一下上述代码运行完,在内存的角度观察他们的变化:

5c0e72ed051c4577ac7b1208bfc042b3.jpg

能够看的出,总共分为三个部分:上面两个框分别是B和C类的,谁先继承谁就在上面,并且,数据a 也是独立开来的,B里面a = 2;C里面a = 6 。

再来看加了virtual后的又会是什么样的变化:

7cef93b9c8d44c838040c0d88c1291a0.jpg

同样也是分为了三个部分,先继承的类就在上面,所以上面两个分别是B和C类,但这次不一样的是:


数据a并没有分开来,而是在最下面去了,相代替的是一个指针,这个指针指向的是一个虚基表,这个表里面存放着一个偏移量,当我们要进行 切割 的时候,获取 数据a 就通过当前的这个指针 + 这个偏离量就能找到了。(这个表叫虚基表,指针叫虚基表指针)


如果此时定义一个B对象,它的里面同时也有一个续集表指针,指向一个自己的续集表,这个偏移量 和 D定义出来的 对象 里的 B类的虚基表里的偏离量不同 

82d2a27a45cf4a3e8c79c3a6caada71e.jpg如图所示:左边是B类定义出来的对象;右边是D类定义出来的对象。

他们都有各自的虚基表指针,这个指针是不一样的!!!!里面的偏移量也是!!

目录
相关文章
|
6月前
虚继承解决菱形继承的原理
虚继承解决菱形继承的原理
|
1月前
|
编译器 C++
C++入门11——详解C++继承(菱形继承与虚拟继承)-2
C++入门11——详解C++继承(菱形继承与虚拟继承)-2
30 0
|
1月前
|
程序员 C++
C++入门11——详解C++继承(菱形继承与虚拟继承)-1
C++入门11——详解C++继承(菱形继承与虚拟继承)-1
32 0
|
4月前
|
编译器 C++ 开发者
C++一分钟之-多重继承与菱形问题
【7月更文挑战第19天】C++的多重继承允许类从多个基类派生,但引入了菱形问题,即类D通过B和C(都继承自A)双重继承A,可能导致数据冗余和二义性。解决这个问题的关键是**虚继承**,通过`virtual`关键字确保基类A只被继承一次,消除冲突。理解并适当使用虚继承是处理这类问题的关键,有助于保持代码的清晰和正确性。
66 0
|
5月前
|
C++
C++一分钟之-继承与多态概念
【6月更文挑战第21天】**C++的继承与多态概述:** - 继承允许类从基类复用代码,增强代码结构和重用性。 - 多态通过虚函数实现,使不同类对象能以同一类型处理。 - 关键点包括访问权限、构造/析构、菱形问题、虚函数与动态绑定。 - 示例代码展示如何创建派生类和调用虚函数。 - 注意构造函数初始化、空指针检查和避免切片问题。 - 应用这些概念能提升程序设计和维护效率。
39 2
|
6月前
|
安全 程序员 编译器
【C++】继承(定义、菱形继承、虚拟继承)
【C++】继承(定义、菱形继承、虚拟继承)
65 1
|
6月前
|
程序员 编译器 C++
【继承】菱形继承以及虚拟菱形继承
【继承】菱形继承以及虚拟菱形继承
|
存储 C++
C++中菱形继承中继承不明确问题
C++中菱形继承中继承不明确问题
75 0
|
安全 程序员 编译器
C++中的继承/虚继承原理
C++中的继承/虚继承原理
111 0
|
6月前
|
C++
C++继承、多继承及菱形继承
C++继承、多继承及菱形继承