該文為轉(zhuǎn)載.原文來(lái)自
http://baike.baidu.com/view/161302.htm1. 虛函數(shù)的定義
虛函數(shù)用來(lái)表現(xiàn)基類和派生類的成員函數(shù)之間的一種關(guān)系.
虛函數(shù)的定義在基類中進(jìn)行,在需要定義為虛函數(shù)的成員函數(shù)的聲明前冠以關(guān)鍵字 virtual.
基類中的某個(gè)成員函數(shù)被聲明為虛函數(shù)后,此虛函數(shù)就可以在一個(gè)或多個(gè)派生類中被重新定義.
在派生類中重新定義時(shí),其函數(shù)原型,包括返回類型,函數(shù)名,參數(shù)個(gè)數(shù),參數(shù)類型及參數(shù)的先后順序,都必須與基類中的原型完全相同.
虛函數(shù)是重載的一種表現(xiàn)形式,是一種動(dòng)態(tài)的重載方式.
2. 為什么使用虛函數(shù)
#include
class CBase{
public:
void who( )
{cout<<"this is the base class!\n";}
};
class CDerive1 : public CBase{
public:
void who( )
{cout<<"this is the derive1 class!\n";}
};
class CDerive2 : public CBase{
public:
void who( )
{cout< who( );
p = &obj2;
p -> who( );
p = &obj3;
p -> who( );
obj2.who( );
obj3.who( );
return 1;
}
運(yùn)行結(jié)果:
this is the base class!
this is the base class!
this is the base class!
this is the derive1 class!
this is the derive2 class!
通過(guò)對(duì)象指針進(jìn)行的普通成員函數(shù)調(diào)用,僅僅與指針的類型有關(guān),而與此刻正指向什么對(duì)象無(wú)關(guān).要想實(shí)現(xiàn)當(dāng)指針指向不同對(duì)象時(shí)執(zhí)行不同的操作,就必須將基類相應(yīng)中的成員函數(shù)定義為虛函數(shù).
3. 虛函數(shù)與重載函數(shù)的關(guān)系
一般的重載函數(shù),函數(shù)的返回類型及所帶的參數(shù)必須至少有一樣不完全相同,只需函數(shù)名相同即可.
基類中定義的虛函數(shù)在派生類中重新定義時(shí),其函數(shù)原型,包括返回類型,函數(shù)名,參數(shù)個(gè)數(shù),參數(shù)類型及參數(shù)的先后順序,都必須與基類中的原型完全相同.
重載虛函數(shù)時(shí),若與基類中的函數(shù)原型出現(xiàn)不同,系統(tǒng)將根據(jù)不同情況分別處理:
(1)僅僅返回類型不同,其余相同,系統(tǒng)會(huì)當(dāng)作出錯(cuò)處理;
(2)函數(shù)原型不同,僅僅函數(shù)名相同,系統(tǒng)會(huì)認(rèn)為是一般的函數(shù)重載,將丟失虛特性.
3.3.4 虛基類
#include
class x{
protected:
int a;
public:
void f ( ) ;
};
class x1 : public x {
public:
x1( ){cout<};
class x2 : public x {
public:
x2( ){ cout<};
class y : public x1, public x2{
public:
y( ){ cout<};
main( )
{
y obj; //error
obj . f ( ) ; //error
return ;
}
二義性錯(cuò)誤
非虛基類的類層次
虛基類的類層次
當(dāng)在多條繼承路徑上有一個(gè)公共的基類,在這些路徑中的某幾條匯合處,這個(gè)公共的基類就會(huì)產(chǎn)生多個(gè)實(shí)例(或多個(gè)副本),若只想保存這個(gè)基類的一個(gè)實(shí)例,可以將這個(gè)公共基類說(shuō)明為虛基類.
class x1 : virtual public x
{
// … …
};
class x2 : virtual public x
{
// … …
};
虛基類的初始化
虛基類的初始化與一般多繼承的初始化在語(yǔ)法上是一樣的,但構(gòu)造函數(shù)的調(diào)用次序不同.
派生類構(gòu)造函數(shù)的調(diào)用次序有三個(gè)原則:
(1) 虛基類的構(gòu)造函數(shù)在非虛基類之前調(diào)用;
(2) 若同一層次中包含多個(gè)虛基類,這些虛基類的構(gòu)造函數(shù)按它們說(shuō)明的次序調(diào)用;
(3) 若虛基類由非虛基類派生而來(lái),則仍先調(diào)用基類構(gòu)造函數(shù),再調(diào)用派生類的構(gòu)造函數(shù).
純虛函數(shù)
僅僅用來(lái)為要從基類中派生的函數(shù)占據(jù)一個(gè)位置。
純虛函數(shù)在基類中沒(méi)有定義,它們被初始化為0。
任何用純虛函數(shù)派生的類,都要自己提供該函數(shù)的具體實(shí)現(xiàn)。
定義純虛函數(shù)
virtual void myMethod(void) = 0;