菱形继承和虚继承

本文介绍了单继承和多继承的概念,重点讨论了菱形继承带来的数据冗余和二义性问题,并阐述了虚继承如何通过使用偏移量解决这些问题。然而,虚继承虽然减少了冗余,但也带来了性能损耗。最后,提到了友元关系不继承以及静态成员在继承体系中的特性。

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

一、单继承和多继承

  • 单继承:一个子类只有一个直接父类时,称这个继承为单继承。
  • 多继承:一个子类有俩个或以上的直接父类时称这个继承关系为多继承。

单继承:可以有多个父类,但是只能由一个直接父类。示意图如下:

多继承示意图如下:

菱形继承示意图如下:

菱形继承对象模型:Assistant的对象中有两份Person成员,菱形继承存在数据冗余和二义性的问题。

菱形继承举例:

class person
{
public:
	string _name;
};
 
class student : public person
{
protected:
	int _num;
};
 
class teather :public person
{
protected:
	int _id;
};
 
class Assistant :public student, public teather
{
protected:
	string _majorCourse;
};
 
void Test()
{
	Assistant a;
	//a._name = "xxx";    //对_name的访问不明确,要指定他们在哪个类里
	a.student::_name = "xxx";
	a.teather::_name = "yyy";
}

这个编译是可以通过的,但是我们发现这样的话,一个人就具有了俩个名字,这样的菱形继承是被人诟病的。说明我们的菱形继承存在二义性和数据冗余问题。

三、虚继承(virtual)——解决菱形继承的二义性和数据冗余的问题

  • 虚继承解决了在菱形继承体系里面子类对象包含多份父类对象的数据冗余和浪费空间的问题。
  • 虚继承体系在实际应用中,我们通常不会定义这么复杂的继承体系。一般不要定义菱形结构的虚继承体系,因为使用虚继承解决数据冗余问题的同时也带来了性能上的损耗。

菱形虚拟继承对象模型:图中解决二义性时在vs下使用的是偏移量。

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;
};
 
void Test()
{
	D dd;
	cout << sizeof(dd) << endl;   //24
	dd.B::_a = 1;
	dd._b = 3;
	dd.C::_a = 2;
	dd._c = 4;
	dd._d = 5;
 
}

 虚继承到底做了什么?

虚继承之所以能解决二义性和数据冗余的问题,他们在原本需要存放数据的地方存放了一个地址,而真正的“_a”数据放在对象的最底下,而我们只要通过这个地址里的内容——偏移量,就能找到我们实际要访问的变量“_a”。

NOTE:

友元关系不能继承,也就是说友元不能访问子类私有和保护成员。

父类定义了static成员,则整个继承体系里面只要一个这样的成员,无论子类生出多少个子类,都只有一个static成员实例。

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值