C++中的继承(三)
C++中的繼承(三)
文章目錄
- C++中的繼承(三)
- 一、復雜菱形繼承和菱形虛擬繼承
- 二、菱形繼承的問題:
- 三、解決菱形繼承
- 四、注意
一、復雜菱形繼承和菱形虛擬繼承
1.單繼承:一個子類只有一個直接父類時稱這種繼承關系為單繼承
2.多繼承:一個子類有兩個或兩個以上的直接父類時,稱這種繼承關系為多繼承。
3.菱形繼承:為多繼承的一種特殊情況
二、菱形繼承的問題:
1.先看一段代碼
class Person { public :string _name ; // 姓名 };class Student : public Person { protected :int _num ; //學號 };class Teacher : public Person { protected :int _id ; // 職工編號 };class Assistant : public Student, public Teacher { protected :string _majorCourse ; // 主修課程 };void Test () {// 這樣會有二義性無法明確知道訪問的是哪一個Assistant a ;a._name = "peter";// 需要顯示指定訪問哪個父類的成員可以解決二義性問題,但是數據冗余問題無法解決a.Student::_name = "xxx";a.Teacher::_name = "yyy"; }通過上面的代碼就可以發現,菱形繼承的問題是容易出現二義性
即person類中有一個成員變量name,student類和teacher類都繼承person類,那么student和teacher都有person類中的成員變量name
那么,Assistant類繼承student和teacher類,就說明類中既有student中的name,也有teacher類中的name
此時如果在Assistant類中訪問name成員變量,就會出現二義性,不知道訪問那個name,此時就得加類名來訪問避免二義興。
三、解決菱形繼承
1.采用菱形虛擬繼承來解決菱形繼承的二義性:
class Person { public :string _name ; // 姓名 };class Student : virtual public Person { protected :int _num ; //學號 };class Teacher : virtual public Person { protected :int _id ; // 職工編號 };class Assistant : public Student, public Teacher { protected :string _majorCourse ; // 主修課程 };void Test () {Assistant a ;a._name = "peter"; }2.解決的原理:
用下面的代碼來研究菱形虛擬繼承的原理:
class A { public:int _a; };// class B : public A class B : virtual public A { public:int _b; };// class C : public A class C : virtual public A { public:int _c; };class D : public B, public C { public:int _d; };int main() {D d;d.B::_a = 1;d.C::_a = 2;d._b = 3;d._c = 4;d._d = 5;return 0; }結合內存看一下:
菱形繼承:
從上面很容易看出菱形繼承的冗余
菱形虛擬繼承:
結合自己畫的對象模型:
菱形繼承:
菱形虛擬繼承:
總結:結合內存圖和對象模型可以得出結論:
在派生類中,在D對象中將A對象放到派生類D的最下面,B和C取4個字節用來存放虛基表指針,虛基表指針指向虛基表,虛基表中存放偏移量,虛基表中第一個存放想對自己的偏移量,第二個存放相對_a的偏移量,通過偏移量來找到_a,這樣使得A類中的數據只存一份,就解決了菱形繼承的二義性
四、注意
- 1.注意菱形虛擬繼承的sizeof大小;
答案是C
總結
以上是生活随笔為你收集整理的C++中的继承(三)的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 进程间程序替换和minishell
- 下一篇: C++中的多态(一)