#include <iostream>
using namespace std;
/*
菱形继承概念:
两个派生类继承同一个基类
又有某个类同时继承这两个派生类
这种继承被称为菱形继承或者钻石继承
菱形继承经典案例
动物类(基类)-----羊(子类)
| |
| |
骆驼(子类)------羊驼(孙子类)
菱形继承问题:
1、羊继承了动物的数据,骆驼同样继承了动物的数据,当羊驼使用继承的数据时,会产生二义性
2、羊驼继承自动物的数据继承了两份(两份分别来自羊和骆驼),其实我们应该清除,这份数据我们只需要一份就可以。
*/
//动物类
class Animal
{
public :
int m_Age;
};
//羊类
class Sheep:virtual public Animal{};
//骆驼类
class Tuo :virtual public Animal{};
//羊驼类
class SheepTuo :public Sheep, public Tuo{};
void test01()
{
SheepTuo st;
//当菱形继承时,两个父类拥有相同数据,需要加作用域区分
st.Sheep::m_Age = 18;
st.Tuo::m_Age = 28;
cout << " st.Sheep::m_Age = " << st.Sheep::m_Age << endl;
cout << " st.Tuo::m_Age = " << st.Tuo::m_Age << endl;
//这份数据我们知道,只要有一份就可以,菱形继承导致数据有两份,资源浪费
//利用虚继承,可以解决菱形继承的问题
// 在继承前,加上关键字 virtual 变为虚继承
// Animal类 称为 虚基类
//此时可以用st直接访问m_Age
cout << " st.m_Age = " << st.m_Age << endl;
}
int main()
{
test01();
return 0;
}
虚基类说明
借助Developer Command Prompt for VS 2022,跳转到源码所在路径,输入如下:
#cl /d1 reportSingleClassLayout类名 程序文件名.cpp
cl /d1 reportSingleClassLayoutSheepTuo 菱形继承.cpp
基类:Animal
子类:Sheep,Tuo
孙子类:SheepTuo
在继承时,加virtual,将Animal标记为虚基类,此时Sheep和Tuo类会分别生成一个虚基类指针(vbptr),这两个虚基类指针会分别指向两个虚基类表(vbtable)
在上图中,Sheep的虚基类指针(起始地址为0)指向Sheep的虚基类表,虚基类表中指向偏移为8,则最终指向地址为0+8(Animal.m_Age
)
同理,Tuo的虚基类指针(起始地址为4)指向Tuo的虚基类表,虚基类表中指向偏移为4,则最终指向地址为4+4(Animal.m_Age
)