C++的精髓——虛函數(shù)
純虛函數(shù)在基類中是沒(méi)有定義的,必須在子類中加以實(shí)現(xiàn),很像java中的接口函數(shù)!
虛函數(shù)
引入原因:為了方便使用多態(tài)特性,我們常常需要在基類中定義虛函數(shù)。
class Cman
{
public:
virtual void Eat(){……};
void Move();
private:
};
class CChild : public CMan
{
public:
virtual void Eat(){……};
private:
};
CMan m_man;
CChild m_child;
//這才是使用的精髓,如果不定義基類的指針去使用,沒(méi)有太大的意義
CMan *p ;
p = &m_man ;
p->Eat(); //始終調(diào)用CMan的Eat成員函數(shù),不會(huì)調(diào)用?CChild?的
p = &m_child;
p->Eat(); //如果子類實(shí)現(xiàn)(覆蓋)了該方法,則始終調(diào)用CChild的Eat函數(shù)
//不會(huì)調(diào)用CMan?的?Eat?方法;如果子類沒(méi)有實(shí)現(xiàn)該函數(shù),則調(diào)用CMan的Eat函數(shù)
p->Move(); //子類中沒(méi)有該成員函數(shù),所以調(diào)用的是基類中的
純虛函數(shù)
引入原因:
1、同“虛函數(shù)”;
2、在很多情況下,基類本身生成對(duì)象是不合情理的。例如,動(dòng)物作為一個(gè)基類可以派生出老虎、孔雀等子類,但動(dòng)物本身生成對(duì)象明顯不合常理。
//純虛函數(shù)就是基類只定義了函數(shù)體,沒(méi)有實(shí)現(xiàn)過(guò)程定義方法如下
//?virtual void Eat() = 0;?直接=0?不要?在cpp中定義就可以了
//純虛函數(shù)相當(dāng)于接口,不能直接實(shí)例話,需要派生類來(lái)實(shí)現(xiàn)函數(shù)定義
//有的人可能在想,定義這些有什么用啊?,我覺(jué)得很有用
//比如你想描述一些事物的屬性給別人,而自己不想去實(shí)現(xiàn),就可以定
//義為純虛函數(shù)。說(shuō)的再透徹一些。比如蓋樓房,你是老板,你給建筑公司
//描述清楚你的樓房的特性,多少層,樓頂要有個(gè)花園什么的
//建筑公司就可以按照你的方法去實(shí)現(xiàn)了,如果你不說(shuō)清楚這些,可能建筑
//公司不太了解你需要樓房的特性。用純需函數(shù)就可以很好的分工合作了
虛函數(shù)和純虛函數(shù)區(qū)別
觀點(diǎn)一:
類里聲明為虛函數(shù)的話,這個(gè)函數(shù)是實(shí)現(xiàn)的,哪怕是空實(shí)現(xiàn),它的作用就是為了能讓這個(gè)函數(shù)在它的子類里面可以被重載,這樣的話,這樣編譯器就可以使用后期綁定來(lái)達(dá)到多態(tài)了
純虛函數(shù)只是一個(gè)接口,是個(gè)函數(shù)的聲明而已,它要留到子類里去實(shí)現(xiàn)。
class A{
protected:
void foo();//普通類函數(shù)
virtual void foo1();//虛函數(shù)
virtual void foo2() = 0;//純虛函數(shù)
}
觀點(diǎn)二:
虛函數(shù)在子類里面也可以不重載的;但純虛必須在子類去實(shí)現(xiàn),這就像Java的接口一樣。通常我們把很多函數(shù)加上virtual,是一個(gè)好的習(xí)慣,雖然犧牲了一些性能,但是增加了面向?qū)ο蟮亩鄳B(tài)性,因?yàn)槟愫茈y預(yù)料到父類里面的這個(gè)函數(shù)不在子類里面不去修改它的實(shí)現(xiàn)
觀點(diǎn)三:
虛函數(shù)的類用于“實(shí)作繼承”,繼承接口的同時(shí)也繼承了父類的實(shí)現(xiàn)。當(dāng)然我們也可以完成自己的實(shí)現(xiàn)。純虛函數(shù)的類用于“介面繼承”,主要用于通信協(xié)議方面。關(guān)注的是接口的統(tǒng)一性,實(shí)現(xiàn)由子類完成。一般來(lái)說(shuō),介面類中只有純虛函數(shù)的。
觀點(diǎn)四:
錯(cuò)誤:帶純虛函數(shù)的類叫虛基類,這種基類不能直接生成對(duì)象,而只有被繼承,并重寫其虛函數(shù)后,才能使用。這樣的類也叫抽象類。
虛函數(shù)是為了繼承接口和默認(rèn)行為
純虛函數(shù)只是繼承接口,行為必須重新定義
////////////////////////////////////////////////////////////////////////////////////
虛基類的初始化 虛基類的初始化與一般多繼承的初始化在語(yǔ)法上是一樣的,但構(gòu)造函數(shù)的調(diào)用次序不同.
派生類構(gòu)造函數(shù)的調(diào)用次序有三個(gè)原則: (1)虛基類的構(gòu)造函數(shù)在非虛基類之前調(diào)用; (2)若同一層次中包含多個(gè)虛基類,這些虛基類的構(gòu)造函 ??虛基類和非虛基類的區(qū)別
數(shù)按它們說(shuō)明的次序調(diào)用; (3)若虛基類由非虛基類派生而來(lái),則仍先調(diào)用基類構(gòu)造函數(shù),再調(diào)用派生類的構(gòu)造函數(shù).編輯本段C++的虛基類 在派生類繼承基類時(shí),加上一個(gè)virtual關(guān)鍵詞則為虛擬基類繼承,如: class derive:virtual public base { }; 虛基類主要解決在多重繼承時(shí),基類可能被多次繼承,虛基類主要提供一個(gè)基類給派生類,如: class B { }; class D1:public B { }; class D2:public B { }; class C:public D1,public D2 { }; 這里C在D1,D2上繼承,但有兩個(gè)基類,造成混亂。因而使用虛基類,即: classB { }; class D1:virtual public B { }; class D2:virtual publicB { }; class C:public D1,public D2編輯本段在使用虛基類時(shí)要注意: (1) 一個(gè)類可以在一個(gè)類族中既被用作虛基類,也被用作非虛基類。 (2) 在派生類的對(duì)象中,同名的虛基類只產(chǎn)生一個(gè)虛基類子對(duì)象,而某個(gè)非虛基類產(chǎn)生各自的子對(duì)象。 (3) 虛基類子對(duì)象是由最遠(yuǎn)派生類的構(gòu)造函數(shù)通過(guò)調(diào)用虛基類的構(gòu)造函數(shù)進(jìn)行初始化的?! ?4) 最遠(yuǎn)派生類是指在繼承結(jié)構(gòu)中建立對(duì)象時(shí)所指定的類?! ?5) 派生類的構(gòu)造函數(shù)的成員初始化列表中必須列出對(duì)虛基類構(gòu)造函數(shù)的調(diào)用;如果未列出,則表示使用該虛基類的缺省構(gòu)造函數(shù)?! ?6) 從虛基類直接或間接派生的派生類中的構(gòu)造函數(shù)的成員初始化列表中都要列出對(duì)虛基類構(gòu)造函數(shù)的調(diào)用。但僅僅用建立對(duì)象的最遠(yuǎn)派生類的構(gòu)造函數(shù)調(diào)用虛基類的構(gòu)造函數(shù),而該派生類的所有基類中列出的對(duì)虛基類的構(gòu)造函數(shù)的調(diào)用在執(zhí)行中被忽略,從而保證對(duì)虛基類子對(duì)象只初始化一次?! ?7) 在一個(gè)成員初始化列表中同時(shí)出現(xiàn)對(duì)虛基類和非虛基類構(gòu)造函數(shù)的調(diào)用時(shí),虛基類的構(gòu)造函數(shù)先于非虛基類的構(gòu)造函數(shù)執(zhí)行。靜態(tài)聯(lián)編:在程序鏈接階段就可以確定的調(diào)用。
動(dòng)態(tài)聯(lián)編:在程序執(zhí)行時(shí)才能確定的調(diào)用。