C++菱形继承

本文介绍了虚继承的概念,旨在解决C++中的菱形继承问题。在多继承的场景下,虚继承确保所有派生类对同一基类的数据只有一份拷贝,避免了数据冗余和二义性。通过案例展示了如何使用虚继承来初始化和访问虚基类的成员。
摘要由CSDN通过智能技术生成

目录

简介

虚继承解决菱形继承问题


简介

两个派生类继承同一个基类,又有某个类同时继承了两个派生类,这种继承称为菱形继承或者钻石继承。

 

菱形继承产生的问题:

设基类为A ,继承基类的两个子类为B、C,有D同时继承B、C

1.B、C都继承了A的数据,当D使用数据时会产生二义性

2.D继承了A的两份数据,我们只需要一份

虚继承解决菱形继承问题

关于虚继承的几点注意事项

1.在多继承情况下,虚基类关键字virtual只对紧跟其后的基类起作用。声明了虚基类之后,虚基类的成员在进一步派生过程中和派生类一起维护同一个内存数据拷贝。

2.建立对象时所指定的类称为最(远)派生类。虚基类成员是由最派生类的构造函数通过调用虚基类的构造函数进行初始化的。

3.在整个继承结构中,直接或间接继承虚基类的所有派生类,都必须在构造函数的成员初始化列表中给出对虚基类的构造函数的调用。如果未列出,则表示调用该虚基类的缺省构造函数。在建立对象时,只有最派生类的构造函数调用虚基类的构造函数,该派生类的其它基类对虚基类构造函数的调用被忽略。

案例

#include<iostream> 
using namespace std; 
class B0
{
public:
	B0(int n) {
		nV = n;
	}
	int nV;
	void fun() {
		cout << "Memeber of B0" << endl;
	}
};
class B1 :virtual public B0
{
public:
	B1(int a):B0(a){}
	int nV1;
};
class B2 :virtual public B0
{
public:
	B2(int a):B0(a){}
	int nV2;
};
class D1 :public B1, public B2
{
public:
	D1(int a):B0(a),B1(a),B2(a){}
	int nVd;
	void fund() {
		cout << "Memver of D1" << endl;
	}
};
int main() 
{  
	D1 d(1);
	cout << d.nV << endl;
	d.fun();
	return 0; 
}

菱形继承指的是一个派生类继承自两个直接或间接基类,而这两个基类又共同继承自一个共同的基类,导致派生类存在两份共同基类的数据成员,从而产生了命名冲突和二义性的问题。 解决菱形继承问题的一种方法是使用虚拟继承。虚拟继承可以使得共同基类派生类只有一份实例,从而避免了数据成员的重复和命名冲突问题。在使用虚拟继承,需要在继承语句前加上关键字 virtual,例如: ``` 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; }; ``` 在上面的例子,B 和 C 都虚拟继承自 A,而 D 继承自 B 和 C。因此,在 D 只有一份 A 的实例,从而避免了数据成员的重复和命名冲突问题。 在初始化菱形继承派生类,需要注意以下几点: 1. 派生类的构造函数必须调用每个直接基类的构造函数,以及虚拟基类的构造函数,顺序为先虚拟基类,再按照继承的顺序调用直接基类的构造函数。 2. 虚拟基类的构造函数由最底层的派生类负责调用,其他派生类不需要再次调用虚拟基类的构造函数。 3. 派生类的析构函数必须调用每个直接基类的析构函数,以及虚拟基类的析构函数,顺序为先按照继承的顺序调用直接基类的析构函数,再调用虚拟基类的析构函数。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值