C++中的菱形继承【详细讲解】

1、菱形继承实例

骡子的产生

2、菱形继承的概念

菱形继承也称为钻石继承。 类D继承自类B和类C,而类B和类C共同继承自类A,这就形成了菱形继承。

3、菱形继承导致的问题

菱形继承会产生二义性的问题,并且会浪费内存空间

实例:

#include <iostream>

using namespace std;
//马科
class Equidae {
public:
	int Age;
};
//马
class Horse :public Equidae{
public:

};
//驴
class Donkey :public Equidae{
public:

};
//骡子
class Mule :public Horse, public Donkey {
public:

};

int main()
{
	class Mule mule;
	cout << "骡子派生类的大小为:" << sizeof(class Mule) << endl;
	mule.Age = 18;
	return 0;
}

运行结果:

结果解释:

由于采用了菱形继承,因此会产生二义性问题。同时,上述程序中骡子派生类的大小为8字节,也造成了内存空间的浪费

4、如何解决菱形继承产生的问题--【虚继承】

1)基本概念

通过使用虚继承来【解决从不同途径继承来的同名的成员属性在内存中由不同的拷贝造成数据不一致的问题】,将共同基类设置为【虚基类】。设置完成以后从不同路径继承过来的同名成员属性在内存中就只有一个拷贝同名成员函数也只有一个映射

  • 虚基类:在继承时,在继承方式的前面加上一个virtual关键字即可;

  • 虚基表:每个虚继承的子类都有一个虚基类指针【是指针,具体所占字节大小看平台】虚基类表【不占用类对象内存空间】。【注意】虚基类依旧会在子类里面存在拷贝,但是仅仅是多存在一份而已,并不在子类中了。当虚继承的子类被当做父类继承时(比如上述举例中的马和驴),虚基类指针也会被继承。

2)关键字

关键字:virtual
实例:class Horse : virtual public Equidae{};

3)实例

#include <iostream>

using namespace std;
//马科
class Equidae {
public:
	int Age;
};
//马
class Horse : virtual public Equidae{    //进行虚继承,避免二义性的问题
public:

};
//驴
class Donkey : virtual public Equidae{   //进行虚继承,避免二义性的问题
public:

};
//骡子
class Mule :public Horse, public Donkey {
public:

};

int main()
{
	class Mule mule;
	cout << "马科的大小为:" << sizeof(Equidae) << endl;
	cout << "马类的大小为:" << sizeof(Horse) << endl;
	cout << "驴类的大小为:" << sizeof(Donkey) << endl;
	cout << "骡子派生类的大小为:" << sizeof(class Mule) << endl;   //这里内存比原来的菱形继承变大了4个字节,是因为虚基表指针的
	mule.Age = 10;   //进行了虚继承,这里就不会出现指代不明确的问题。
	return 0;
}

 运行结果:【Visual Studio工具下运行结果】

马科的大小为:4
马类的大小为:16
驴类的大小为:16
骡子派生类的大小为:24

 运行结果:【32位linux下g++编译执行的结果】

马科的大小为:4
马类的大小为:8
驴类的大小为:8
骡子派生类的大小为:12

结果解释:

  • 两种编译的标准不同,因此产生的结果不同。
  • 对32位linux下g++编译执行的结果进行解释:

5、总结

在编写面向对象程序时,能使用单一继承尽量不使用多继承。首先,可能会造成成员名称重复的问题。其次,可能会产生菱形继承的问题,导致二义性和内存浪费。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
菱形继承的优点: 1. 代码重用:菱形继承允许子类同时继承两个不同的父类,从而可以重用这两个父类的成员和方法,减少了代码的重复编写。 2. 灵活性:菱形继承可以实现多继承的一种方式,使得子类可以同时具有多个父类的特性和功能,增加了灵活性和扩展性。 3. 层次清晰:菱形继承的结构相对清晰,可以更好地表示对象之间的关系和继承层次。 菱形继承的缺点: 1. 冗余数据:由于派生类同时继承两个父类,而这两个父类又都继承自同一个基类,导致派生类可能存在相同的成员变量和方法,造成了数据的冗余。 2. 二义性:菱形继承可能引发成员函数的二义性问题。如果两个父类存在相同的成员函数,而派生类又没有明确指定使用哪个父类的成员函数,会导致编译器无法确定调用哪个函数。 3. 破坏封装性:菱形继承可能破坏基类的封装性。如果两个父类都访问了基类的私有成员,而派生类又同时继承了这两个父类,就会导致派生类可以直接访问基类的私有成员,违背了封装的原则。 为了解决菱形继承带来的冗余数据和二义性问题,C++ 提供了虚继承(virtual inheritance)的机制。通过在派生类对基类的继承声明使用 `virtual` 关键字,可以避免数据冗余和二义性的问题。虚继承将共同基类的成员在派生类只保留一份,解决了数据冗余的问题;而对于二义性问题,则需要在派生类明确指定调用哪个父类的成员函数。 综上所述,菱形继承在代码重用和灵活性方面具有优点,但也存在冗余数据、二义性和破坏封装性的缺点。在使用菱形继承时,需要注意这些问题,并根据具体情况选择使用虚继承或其他方式来解决。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值