第七章:圖文例解C++類的多重繼承與虛擬繼承

字號:

在過去的學習中,我們始終接觸的單個類的繼承,但是在現(xiàn)實生活中,一些新事物往往會擁有兩個或者兩個以上事物的屬性,為了解決這個問題,C++引入了多重繼承的概念,C++允許為一個派生類指定多個基類,這樣的繼承結構被稱做多重繼承。
    舉個例子,交通工具類可以派生出汽車和船連個子類,但擁有汽車和船共同特性水陸兩用汽車就必須繼承來自汽車類與船類的共同屬性。
    由此我們不難想出如下的圖例與代碼:
    當一個派生類要使用多重繼承的時候,必須在派生類名和冒號之后列出所有基類的類名,并用逗好分隔。
    #include
    using namespace std;
    class Vehicle
    {
    public:
    Vehicle(int weight = 0)
    {
    Vehicle::weight = weight;
    }
    void SetWeight(int weight)
    {
    cout<<"重新設置重量"< Vehicle::weight = weight;
    }
    virtual void ShowMe() = 0;
    protected:
    int weight;
    };
    class Car:public Vehicle//汽車
    {
    public:
    Car(int weight=0,int aird=0):Vehicle(weight)
    {
    Car::aird = aird;
    }
    void ShowMe()
    {
    cout<<"我是汽車!"< }
    protected:
    int aird;
    };
    class Boat:public Vehicle//船
    {
    public:
    Boat(int weight=0,float tonnage=0):Vehicle(weight)
    {
    Boat::tonnage = tonnage;
    }
    void ShowMe()
    {
    cout<<"我是船!"< }
    protected:
    float tonnage;
    };
    class AmphibianCar:public Car,public Boat//水陸兩用汽車,多重繼承的體現(xiàn)
    {
    public:
    AmphibianCar(int weight,int aird,float tonnage)
    :Vehicle(weight),Car(weight,aird),Boat(weight,tonnage)
    //多重繼承要注意調用基類構造函數(shù)
    {
    }
    void ShowMe()
    {
    cout<<"我是水陸兩用汽車!"< }
    };
    int main()
    {
    AmphibianCar a(4,200,1.35f);//錯誤
    a.SetWeight(3);//錯誤
    system("pause");
    }
    上面的代碼從表面看,看不出有明顯的語發(fā)錯誤,但是它是不能夠通過編譯的。這有是為什么呢?
    這是由于多重繼承帶來的繼承的模糊性帶來的問題。
    先看如下的圖示:
    在圖中深紅色標記出來的地方正是主要問題所在,水陸兩用汽車類繼承了來自Car類與Boat類的屬性與方法,Car類與Boat類同為AmphibianCar類的基類,在內存分配上AmphibianCar獲得了來自兩個類的SetWeight()成員函數(shù),當我們調用a.SetWeight(3)的時候計算機不知道如何選擇分別屬于兩個基類的被重復擁有了的類成員函數(shù)SetWeight()。
    由于這種模糊問題的存在同樣也導致了AmphibianCar a(4,200,1.35f);執(zhí)行失敗,系統(tǒng)會產生Vehicle”不是基或成員的錯誤。
    以上面的代碼為例,我們要想讓AmphibianCar類既獲得一個Vehicle的拷貝,而且又同時共享用Car類與Boat類的數(shù)據成員與成員函數(shù)就必須通過C++所提供的虛擬繼承技術來實現(xiàn)。
    我們在Car類和Boat類繼承Vehicle類出,在前面加上virtual關鍵字就可以實現(xiàn)虛擬繼承,使用虛擬繼承后,當系統(tǒng)碰到多重繼承的時候就會自動先加入一個Vehicle的拷貝,當再次請求一個Vehicle的拷貝的時候就會被忽略,保證繼承類成員函數(shù)的性。
    修改后的代碼如下,注意觀察變化:
    #include
    using namespace std;
    class Vehicle
    {
    public:
    Vehicle(int weight = 0)
    {
    Vehicle::weight = weight;
    cout<<"載入Vehicle類構造函數(shù)"< }
    void SetWeight(int weight)
    {
    cout<<"重新設置重量"< Vehicle::weight = weight;
    }
    virtual void ShowMe() = 0;
    protected:
    int weight;
    };
    class Car:virtual public Vehicle//汽車,這里是虛擬繼承
    {
    public:
    Car(int weight=0,int aird=0):Vehicle(weight)
    {
    Car::aird = aird;
    cout<<"載入Car類構造函數(shù)"< }
    void ShowMe()
    {
    cout<<"我是汽車!"< }
    protected:
    int aird;
    };
    class Boat:virtual public Vehicle//船,這里是虛擬繼承
    {
    public:
    Boat(int weight=0,float tonnage=0):Vehicle(weight)
    {
    Boat::tonnage = tonnage;
    cout<<"載入Boat類構造函數(shù)"< }
    void ShowMe()
    {
    cout<<"我是船!"< }
    protected:
    float tonnage;
    };
    class AmphibianCar:public Car,public Boat//水陸兩用汽車,多重繼承的體現(xiàn)
    {
    public:
    AmphibianCar(int weight,int aird,float tonnage)
    :Vehicle(weight),Car(weight,aird),Boat(weight,tonnage)
    //多重繼承要注意調用基類構造函數(shù)
    {
    cout<<"載入AmphibianCar類構造函數(shù)"< }
    void ShowMe()
    {
    cout<<"我是水陸兩用汽車!"< }
    void ShowMembers()
    {
    cout<<"重量:"< }
    };
    int main()
    {
    AmphibianCar a(4,200,1.35f);
    a.ShowMe();
    a.ShowMembers();
    a.SetWeight(3);
    a.ShowMembers();
    system("pause");
    }
    注意觀察類構造函數(shù)的構造順序。
    雖然說虛擬繼承與虛函數(shù)有一定相似的地方,但讀者務必要記住,他們之間是絕對沒有任何聯(lián)系的!