在過去的學(xué)習(xí)中,我們始終接觸的單個類的繼承,但是在現(xiàn)實生活中,一些新事物往往會擁有兩個或者兩個以上事物的屬性,為了解決這個問題,C++引入了多重繼承的概念,C++允許為一個派生類指定多個基類,這樣的繼承結(jié)構(gòu)被稱做多重繼承。
舉個例子,交通工具類可以派生出汽車和船連個子類,但擁有汽車和船共同特性水陸兩用汽車就必須繼承來自汽車類與船類的共同屬性。
由此我們不難想出如下的圖例與代碼:
當(dāng)一個派生類要使用多重繼承的時候,必須在派生類名和冒號之后列出所有基類的類名,并用逗好分隔。
//程序作者:管寧
//站點:www.cndev-lab.com
//所有稿件均有版權(quán),如要轉(zhuǎn)載,請務(wù)必出處和作者
#include
usingnamespacestd;
classVehicle
{
public:
Vehicle(intweight =0)
{
Vehicle::weight =weight;
}
voidSetWeight(intweight)
{
cout<<"重新設(shè)置重量" }
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//水陸兩用汽車,多重繼承的體現(xiàn)
{
public:
AmphibianCar(intweight,intaird,floattonnage)
:Vehicle(weight),Car(weight,aird),Boat(weight,tonnage)
//多重繼承要注意調(diào)用基類構(gòu)造函數(shù)
{
}
voidShowMe()
{
cout<<"我是水陸兩用汽車!"<}
};
intmain()
{
AmphibianCar a(4,200,1.35f);//錯誤
a.SetWeight(3);//錯誤
system("pause");
}
上面的代碼從表面看,看不出有明顯的語發(fā)錯誤,但是它是不能夠通過編譯的。這有是為什么呢?
這是由于多重繼承帶來的繼承的模糊性帶來的問題。
先看如下的圖示:
在圖中深紅色標(biāo)記出來的地方正是主要問題所在,水陸兩用汽車類繼承了來自Car類與Boat類的屬性與方法,Car類與Boat類同為AmphibianCar類的基類,在內(nèi)存分配上AmphibianCar獲得了來自兩個類的SetWeight()成員函數(shù),當(dāng)我們調(diào)用a.SetWeight(3)的時候計算機不知道如何選擇分別屬于兩個基類的被重復(fù)擁有了的類成員函數(shù)SetWeight()。
由于這種模糊問題的存在同樣也導(dǎo)致了AmphibianCar a(4,200,1.35f);執(zhí)行失敗,系統(tǒng)會產(chǎn)生Vehicle”不是基或成員的錯誤。
舉個例子,交通工具類可以派生出汽車和船連個子類,但擁有汽車和船共同特性水陸兩用汽車就必須繼承來自汽車類與船類的共同屬性。
由此我們不難想出如下的圖例與代碼:
當(dāng)一個派生類要使用多重繼承的時候,必須在派生類名和冒號之后列出所有基類的類名,并用逗好分隔。
//程序作者:管寧
//站點:www.cndev-lab.com
//所有稿件均有版權(quán),如要轉(zhuǎn)載,請務(wù)必出處和作者
#include
usingnamespacestd;
classVehicle
{
public:
Vehicle(intweight =0)
{
Vehicle::weight =weight;
}
voidSetWeight(intweight)
{
cout<<"重新設(shè)置重量"
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//水陸兩用汽車,多重繼承的體現(xiàn)
{
public:
AmphibianCar(intweight,intaird,floattonnage)
:Vehicle(weight),Car(weight,aird),Boat(weight,tonnage)
//多重繼承要注意調(diào)用基類構(gòu)造函數(shù)
{
}
voidShowMe()
{
cout<<"我是水陸兩用汽車!"<}
};
intmain()
{
AmphibianCar a(4,200,1.35f);//錯誤
a.SetWeight(3);//錯誤
system("pause");
}
上面的代碼從表面看,看不出有明顯的語發(fā)錯誤,但是它是不能夠通過編譯的。這有是為什么呢?
這是由于多重繼承帶來的繼承的模糊性帶來的問題。
先看如下的圖示:
在圖中深紅色標(biāo)記出來的地方正是主要問題所在,水陸兩用汽車類繼承了來自Car類與Boat類的屬性與方法,Car類與Boat類同為AmphibianCar類的基類,在內(nèi)存分配上AmphibianCar獲得了來自兩個類的SetWeight()成員函數(shù),當(dāng)我們調(diào)用a.SetWeight(3)的時候計算機不知道如何選擇分別屬于兩個基類的被重復(fù)擁有了的類成員函數(shù)SetWeight()。
由于這種模糊問題的存在同樣也導(dǎo)致了AmphibianCar a(4,200,1.35f);執(zhí)行失敗,系統(tǒng)會產(chǎn)生Vehicle”不是基或成員的錯誤。

