<C++> 深度剖析菱形继承

​​在这里插入图片描述

什么是菱形继承

继承关系形如下图的继承即为菱形继承,或者叫钻石继承。

img
菱形继承的问题:公有继承前提下,如果类A中含有成员变量a,那么类B与类C中都有继承自类A的a,类D中又继承类B与类C,所以类D中这时会有两份a,不仅造成了空间冗余,更造成了访问时的二义性问题。

举个例子,代码如下

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

此时访问d对象中的a需要指定作用域访问,否则会报错,编译器也不知道访问的是哪个a,我们不妨打开内存窗口查看d对象

在这里插入图片描述

整个红框就是d对象的对象模型,蓝框就是d对象中继承自类B的部分,绿框就是d对象继承自类C的部分,其中两个紫框就是两份a,这就是造成数据冗余与访问二义性的问题。

解决方法:在类B与类C继承类A时采用虚继承,即在继承方式前加上关键字“virtual”,代码如下

struct A
{
	int a;
};
struct B : virtual public A
{
	int b;
};
struct C :virtual public A
{
	int c;
};
struct D :public B, public C
{
	int d;
};

这时再去访问类D对象中的a即不会产生二义性问题。


探究底层

为什么数据冗余问题与访问二义性只需要添加一个关键字就可以解决?

我们修改一下代码

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

刚才我们查看了没有使用虚继承时的d对象在内存中的情况,现在我们来一起看一下使用虚继承后的内存窗口

在这里插入图片描述

我们可以看到,只保留了一份a,已用黄框标注,原来B和C中存储a的地方换成了两个地址,我们在内存窗口中看看这两个地址里面有什么
在这里插入图片描述

B中地址存储了28H,C中地址存储了20H,转换成十进制就是40和32,如果我们把这两个数字分别加上B中存储地址的地址与C中存储地址的地址,就会发现刚好都等于变量a的地址,这就说明在虚继承中,继承自顶端的成员变量会被替换成一个地址,而这个地址指向的空间存储的内容刚好是该地址对于成员变量的偏移量,也就是说只保留了一份a,所以在访问时才不会造成二义性的问题,并且也解决了冗余的问题。

在这里插入图片描述

  • 10
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

【 Stack_OverFlow 】

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值