菱形继承

本文探讨了菱形继承问题,包括数据冗余和二义性,并介绍了C++通过虚继承来解决这些问题的原理。通过实例展示了虚继承如何消除数据冗余和二义性,以及虚基表在内存中的作用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

菱形继承

下图展示了一个简单的菱形继承(假设是public方式继承)
在这里插入图片描述

菱形继承的问题

数据冗余和二义性

上图B和C都继承了A,B和C的数据中除了有自己的成员变量外,还有继承自A的成员:_a,此时D同时继承了B和C,那么D中除了自己有的成员变量以外,还包含了继承自B和C的成员变量,此时D中就同时有_b,_c,_d,_a 那么_a究竟是B中的还是C中的呢,而且多继承使得数据大量的重复堆积,使得派生类数据量庞大,这就是菱形继承的数据冗余和二义性的问题

如何解决这个问题

C++中提供了一种解决菱形继承带来的数据冗余和二义性的方法,虚继承
在继承方式前加上关键字:virtual就实现了虚继承

class A
{
public:
	int _a;
};
// class B : public A
class B : virtual public A
{
public:
	int _b;
};
// class C : public A
class C : virtual public A
{
public:
	int _c;
};
class D : public B, public C
{
public:
	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;
}

在这里插入图片描述
从VS的监视窗口可以看出,使用d.B::_a和使用d.C::_a访问的是同一个_a,引入了虚继承确实解决了数据冗余和二义性的问题

实现的原理

实现的原理是虚基表(注意和虚函数表的区分,两者不是一个东西,后者是多态的实现原理)
在这里插入图片描述
从内存中可以看出,d的成员变量的地址中除了存有自己的数据以外,还存有一个指针,这个指针就是虚基表指针,它指向一个虚基表,我们可以通过内存查看这个虚基表中的内容,发现它存储了一个值,这个值其实是从当前虚基表指针位置到_a的偏移量(图中显示的数据是16进制表示的,图中显示14说明偏移量是20个字节)

从_b找_a : 0x010FF6F0 + 20 = 0x010FF704 = _a的地址
从_c找_a : 0x010FF6F8 + 12 = 0x010FF704 = _a的地址

### C++ 菱形继承问题及其虚继承解决方案 #### 什么是菱形继承? 在C++中,当一个类通过多条路径从同一个基类派生出来,就会形成所谓的“菱形继承”。具体来说,如果存在四个类 `A`、`B`、`C` 和 `D`,其中 `B` 和 `C` 都继承自 `A`,而 `D` 同继承自 `B` 和 `C`,那么就形成了一个类似于菱形的结构[^1]。 这种继承方式可能导致两个主要问题: 1. **数据冗余**:由于 `B` 和 `C` 都独立地继承了 `A` 的成员变量和函数,因此在 `D` 类对象中可能会出现两份来自 `A` 的成员副本。 2. **二义性**:当访问 `A` 中定义的成员,编译器无法确定应该调用哪一条路径上的版本(即通过 `B` 或者 `C`),从而引发歧义错误[^4]。 #### 如何解决菱形继承问题? 为了克服上述问题,C++ 提供了一种机制称为“虚拟继承”(Virtual Inheritance)。通过声明某个基类为虚基类,可以确保即使该基类被多次间接继承,最终也只会有一个实例存在于派生类中[^2]。 以下是实现虚拟继承的一个典型例子: ```cpp class A { public: void show() { std::cout << "Class A\n"; } }; // 使用virtual关键字指定A作为虚基类 class B : public virtual A {}; class C : public virtual A {}; class D : public B, public C {}; int main() { D obj; obj.show(); // 不会产生歧义,因为只有一份A的show() } ``` 在这个例子中,尽管 `D` 继承了两次 `A`(一次经由 `B`,另一次经由 `C`),但由于采用了虚拟继承的方式,所以实际上只有一个 `A` 对象嵌套在 `D` 当中。这样既消除了重复的数据存储又解决了可能产生的命名冲突问题[^3]。 值得注意的是,虽然虚拟继承能够有效处理菱形继承带来的挑战,但它同也增了程序复杂性和运行开销。这是因为编译器需要额外的工作来管理虚基类指针并调整内存布局以支持唯一的共享子对象。 综上所述,理解如何运用虚拟继承技术对于构建稳定可靠的C++应用程序非常重要;然而,在实际开发过程中应谨慎考虑是否真的有必要引入复杂的多重继承体系——有候简单的组合模式可能是更好的替代方案。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值