所谓菱形继承就是 两个子类继承同一个父类,同时又有一个孙类继承这两个子类。用图来分析即为菱形形状。
class Animal {
public:
int _age;
};
class sheep : public Animal {
};
class tuo : public Animal {
};
class sheeptuo :public sheep, public tuo {
};
在菱形继承中,使用数据会发生数据的二义性,那我们就要在使用数据时加上作用域。但是这时我们就会发现,同一份数据被继承了两份,造成了不必要的空间浪费。
sheeptuo st;
st.sheep::_age = 18;
st.tuo::_age = 28;
cout << st.tuo::_age << endl;
为了更好的分析单一类的布局,我们使用 Visual Studio 自带的 Develop Command Promt 来分析class sheeptuo的数据。
可以看到这时 class sheeotuo 的size为8。继承了两个父类,并且拥有两个_age。
为了解决菱形继承的问题,我们引入虚继承的概念;
class Animal {
public:
int _age;
};
class sheep :virtual public Animal {
};
class tuo : virtual public Animal {
};
class sheeptuo :public sheep, public tuo {
};
void test() {
sheeptuo st;
st._age = 18;
cout << st._age << endl;
}
在继承的模式前加上 virtual关键词。这时我们再来分析它的类布局
可以看到 base class sheep与base class tuo 中没有了_age数据。替而代之的时vbptr(virtual base pointer) 虚基类指针 这个指针指向 下面的 vbtable(virtual base table)虚基类表,vbtable 会有个偏移量,当数据被调用时,就会加上这个偏移量从而指向一份数据,达到调用_age的目的。于是我们看到在class sheeptuo中只有一份_age数据类型。