问题描述
菱形继承问题本质上是一种多继承问题。比如我要定义一个Animal类,在此基类的基础上衍生出两个派生类Sheep、Tuo,但我又想构造一个SheepTuo类(同时具备Sheep和Tuo的属性)。这样一个子类继承多个父类的问题是多继承问题,如果父类之间有同名的方法或者属性,就会产生二义性。
- 继承多份父类的属性,超成资源的浪费
- 同名属性,产生二义性
#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;
}
编译运行界面