//程序作者:管寧
//站點:www.cndev-lab.com
//所有稿件均有版權,如要轉載,請務必著名出處和作者
#include <iostream>
usingnamespacestd;
classVehicle
{
public:
Vehicle(intweight =0)
{
Vehicle::weight =weight;
}
voidSetWeight(intweight)
{
cout<<"重新設置重量"<Vehicle::weight =weight;
}
virtualvoidShowMe() =0;
protected:
intweight;
};
classCar:publicVehicle//汽車
{
public:
Car(intweight=0,intaird=0):Vehicle(weight)
{
Car::aird =aird;
}
voidShowMe()
{
cout<<"我是汽車!"<}
protected:
intaird;
};
classBoat:publicVehicle//船
{
public:
Boat(intweight=0,floattonnage=0):Vehicle(weight)
{
Boat::tonnage =tonnage;
}
voidShowMe()
{
cout<<"我是船!"<}
protected:
floattonnage;
};
classAmphibianCar:publicCar,publicBoat//水陸兩用汽車,多重繼承的體現
{
public:
AmphibianCar(intweight,intaird,floattonnage)
:Vehicle(weight),Car(weight,aird),Boat(weight,tonnage)
//多重繼承要注意調用基類構造函數
{
}
voidShowMe()
{
cout<<"我是水陸兩用汽車!"<}
};
intmain()
{
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>
usingnamespacestd;
classVehicle
{
public:
Vehicle(intweight =0)
{
Vehicle::weight =weight;
cout<<"載入Vehicle類構造函數"<
voidSetWeight(intweight)
{
cout<<"重新設置重量"<
}
virtualvoidShowMe() =0;
protected:
intweight;
};
classCar:virtualpublicVehicle//汽車,這裡是虛擬繼承
{
public:
Car(intweight=0,intaird=0):Vehicle(weight)
{
Car::aird =aird;
cout<<"載入Car類構造函數"<
voidShowMe()
{
cout<<"我是汽車!"<
protected:
intaird;
};
classBoat:virtualpublicVehicle//船,這裡是虛擬繼承
{
public:
Boat(intweight=0,floattonnage=0):Vehicle(weight)
{
Boat::tonnage =tonnage;
cout<<"載入Boat類構造函數"<
voidShowMe()
{
cout<<"我是船!"<
protected:
floattonnage;
};
classAmphibianCar:publicCar,publicBoat//水陸兩用汽車,多重繼承的體現
{
public:
AmphibianCar(intweight,intaird,floattonnage)
:Vehicle(weight),Car(weight,aird),Boat(weight,tonnage)
//多重繼承要注意調用基類構造函數
{
cout<<"載入AmphibianCar類構造函數"<
voidShowMe()
{
cout<<"我是水陸兩用汽車!"<
voidShowMembers()
{
cout<<"重量:"<
intmain()
{
AmphibianCar a(4,200,1.35f);
a.ShowMe();
a.ShowMembers();
a.SetWeight(3);
a