圖文例解C++類的多重繼承與虛擬繼承__C++

來源:互聯網
上載者:User
圖文例解C++類的多重繼承與虛擬繼承

在過去的學習中,我們始終接觸的單個類的繼承,但是在現實生活中,一些新事物往往會擁有兩個或者兩個以上事物的屬性,為瞭解決這個問題,C++引入了多重繼承的概念,C++允許為一個衍生類別指定多個基類,這樣的繼承結構被稱做多重繼承

  舉個例子,交通工具類可以派生出汽車和船連個子類,但擁有汽車和船共同特性水陸兩用汽車就必須繼承來自汽車類與船類的共同屬性。 

  由此我們不難想出如下的圖例與代碼:

  當一個衍生類別要使用多重繼承的時候,必須在衍生類別名和冒號之後列出所有基類的類名,並用逗好分隔。

//程式作者:管寧    
//網站:www.cndev-lab.com    
//所有稿件均有著作權,如要轉載,請務必著名出處和作者    
  
#include <iostream>  
using namespace std;  
  
class Vehicle  
{  
    public:  
        Vehicle(int weight = 0)  
        {  
            Vehicle::weight = weight;  
        }  
        void SetWeight(int weight)  
        {  
            cout<<"重新設定重量"<<endl;  
            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<<"我是汽車。"<<endl;  
        }  
    protected:  
        int aird;  
};  
  
class Boat:public Vehicle//船  
{  
    public:  
        Boat(int weight=0,float tonnage=0):Vehicle(weight)  
        {  
            Boat::tonnage = tonnage;  
        }  
        void ShowMe()  
        {  
            cout<<"我是船。"<<endl;  
        }  
    protected:  
        float tonnage;  
};  
  
class AmphibianCar:public Car,public Boat//水陸兩用汽車,多重繼承的體現  
{  
    public:  
        AmphibianCar(int weight,int aird,float tonnage)  
        :Vehicle(weight),Car(weight,aird),Boat(weight,tonnage)  
        //多重繼承要注意調用基類建構函式  
        {  
          
        }  
        void ShowMe()  
        {  
            cout<<"我是水陸兩用汽車。"<<endl;  
        }  
};  
int main()  
{  
    AmphibianCar a(4,200,1.35f);//錯誤  
    a.SetWeight(3);//錯誤  
    system("pause");   
}

  上面的代碼從表面看,看不出有明顯的語發錯誤,但是它是不能夠通過編譯的。這有是為什麼呢。 
  這是由於多重繼承帶來的繼承的模糊性帶來的問題。 先看如下的圖示:

  在圖中深紅色標記出來的地方正是主要問題所在,水陸兩用汽車類繼承了來自Car類與Boat類的屬性與方法,Car類與Boat類同為AmphibianCar類的基類,在記憶體配置上AmphibianCar獲得了來自兩個類的SetWeight()成員函數,當我們調用a.SetWeight(3)的時候電腦不知道如何選擇分別屬於兩個基類的被重複擁有了的類成員函數SetWeight()。

  由於這種模糊問題的存在同樣也導致了AmphibianCar a(4,200,1.35f);執行失敗,系統會產生Vehicle”不是基或成員的錯誤。

  以上面的代碼為例,我們要想讓AmphibianCar類既獲得一個Vehicle的拷貝,而且又同時共用用Car類與Boat類的資料成員與成員函數就必須通過C++所提供的虛擬繼承技術來實現。

  我們在Car類和Boat類繼承Vehicle類出,在前面加上virtual關鍵字就可以實現虛擬繼承,使用虛擬繼承後,當系統碰到多重繼承的時候就會自動先加入一個Vehicle的拷貝,當再次請求一個Vehicle的拷貝的時候就會被忽略,保證繼承類成員函數的唯一性。 

  修改後的代碼如下,注意觀察變化:

//程式作者:管寧    
//網站:www.cndev-lab.com    
//所有稿件均有著作權,如要轉載,請務必著名出處和作者    
  
#include <iostream>  
using namespace std;  
  
class Vehicle  
{  
    public:  
        Vehicle(int weight = 0)  
        {  
            Vehicle::weight = weight;  
            cout<<"載入Vehicle類建構函式"<<endl;  
        }  
        void SetWeight(int weight)  
        {  
            cout<<"重新設定重量"<<endl;  
            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類建構函式"<<endl;  
        }  
        void ShowMe()  
        {  
            cout<<"我是汽車。"<<endl;  
        }  
    protected:  
        int aird;  
};  
  
class Boat:virtual public Vehicle//船,這裡是虛擬繼承  
{  
    public:  
        Boat(int weight=0,float tonnage=0):Vehicle(weight)  
        {  
            Boat::tonnage = tonnage;  
            cout<<"載入Boat類建構函式"<<endl;  
        }  
        void ShowMe()  
        {  
            cout<<"我是船。"<<endl;  
        }  
    protected:  
        float tonnage;  
};  
  
class AmphibianCar:public Car,public Boat//水陸兩用汽車,多重繼承的體現  
{  
    public:  
        AmphibianCar(int weight,int aird,float tonnage)  
        :Vehicle(weight),Car(weight,aird),Boat(weight,tonnage)  
        //多重繼承要注意調用基類建構函式  
        {  
            cout<<"載入AmphibianCar類建構函式"<<endl;  
        }  
        void ShowMe()  
        {  
            cout<<"我是水陸兩用汽車。"<<endl;  
        }  
        void ShowMembers()  
        {  
            cout<<"重量:"<<weight<<"頓,"<<"空氣排量:"<<aird<<"CC,"<<"排水量:"<<tonnage<<"頓"<<endl;  
        }  
};  
int main()  
{  
    AmphibianCar a(4,200,1.35f);  
    a.ShowMe();  
    a.ShowMembers();  
    a.SetWeight(3);  
    a.ShowMembers();  
    system("pause");   
}

  注意觀察類建構函式的構造順序。 

  雖然說虛擬繼承與虛函數有一定相似的地方,但讀者務必要記住,他們之間是絕對沒有任何聯絡的。 http://pcedu.pconline.com.cn/empolder/gj/c/0503/579115_1.html

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.