问题描述

菱形继承问题本质上是一种多继承问题。比如我要定义一个Animal类,在此基类的基础上衍生出两个派生类Sheep、Tuo,但我又想构造一个SheepTuo类(同时具备Sheep和Tuo的属性)。这样一个子类继承多个父类的问题是多继承问题,如果父类之间有同名的方法或者属性,就会产生二义性。

  1. 继承多份父类的属性,超成资源的浪费
  2. 同名属性,产生二义性
#include "iostream"
using namespace std;

class Animal
{//动物 基类
public:
	int m_age;
};

class Sheep:public Animal
{//羊 SheepTuo的父类
};

class Tuo:public Animal
{//驼子 SheepTuo的父类
};

class SheepTuo:public Sheep, public Tuo
{//羊驼(草泥马) 该类生成的对象会同时从 Sheep和Tuo两个父类中继承m_age属性

};

 

避免多继承问题二义性的方法

使用作用域的方式,可以避免二义性。

SheepTuo st;
st.Tuo::m_age = 10;
st.Sheep::m_age = 20;

cout << st.Tuo::m_age << endl;		//通过作用域来访问m_age,避免二义性
cout << st.Sheep::m_age << endl;

 

消除多继承问题二义性的方法

 菱形继承的解决方案是利用虚继承。将父类修饰成为虚基类,子类从父类继承的过程叫做虚继承。可以实现资源的共享,并且消除二义性。C++内部实现是:虚基类指针 --> 查找虚基类表 --> 找到共享的属性or方法。

我的码云:https://gitee.com/hinzer/my-notes-of-C_plus/tree/master/step9/diamondInherit

#include "iostream"

using namespace std;

class Animal
{//动物 基类
public:
	int m_age;
};

class Sheep:virtual public Animal
{//羊 SheepTuo的父类

};

class Tuo:virtual public Animal
{//驼子 SheepTuo的父类

};

class SheepTuo:public Sheep, public Tuo
{//羊驼(草泥马) 

};

int main(int argc, char const *argv[])
{
	SheepTuo st;

	st.Tuo::m_age = 10;
	st.Sheep::m_age = 20;

	cout << st.Tuo::m_age << endl;		//通过作用域来访问m_age,避免二义性
	cout << st.Sheep::m_age << endl;

	cout << st.m_age << endl;			//通过虚继承,消除二义性,并且可以直接访问


	return 0;
}

 

编译运行界面