虛函數(shù)為了重載和多態(tài)的需要岳守,在基類中是有定義的艾杏,即便定義是空,所以子類中可以重寫也可以不寫基類中的此函數(shù)极颓!
純虛函數(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;
CMan *p ;//這才是使用的精髓骇径,如果不定義基類的指針去使用躯肌,沒有太大的意義
p = &m_man ;
p->Eat(); //始終調(diào)用CMan的Eat成員函數(shù),不會調(diào)用 CChild 的
p = &m_child;
p->Eat(); //如果子類實現(xiàn)(覆蓋)了該方法破衔,則始終調(diào)用CChild的Eat函數(shù)
//不會調(diào)用CMan 的 Eat 方法清女;如果子類沒有實現(xiàn)該函數(shù),則調(diào)用CMan的Eat函數(shù)
p->Move(); //子類中沒有該成員函數(shù)晰筛,所以調(diào)用的是基類中的
純虛函數(shù)
引入原因:
1嫡丙、同“虛函數(shù)”;
2读第、在很多情況下迄沫,基類本身生成對象是不合情理的。例如卦方,動物作為一個基類可以派生出老虎羊瘩、孔雀等子類,但動物本身生成對象明顯不合常理盼砍。
純虛函數(shù)就是基類只定義了函數(shù)體尘吗,沒有實現(xiàn)過程,定義方法如: virtual void Eat() = 0; 不要
在cpp中定義浇坐;純虛函數(shù)相當(dāng)于接口睬捶,不能直接實例話,需要派生類來實現(xiàn)函數(shù)定義近刘;
有的人可能在想擒贸,定義這些有什么用啊
臀晃,我覺得很有用,比如你想描述一些事物的屬性給別人介劫,而自己不想去實現(xiàn)徽惋,就可以定義為純虛函數(shù)。說的再透徹一些座韵。比如蓋樓房险绘,你是老板,你給建筑公司描述清楚你的樓房的特性誉碴,多少層宦棺,樓頂要有個花園什么的,建筑公司就可以按照你的方法去實現(xiàn)了黔帕,如果你不說清楚這些代咸,可能建筑公司不太了解你需要樓房的特性。用純需函數(shù)就可以很好的分工合作了
虛函數(shù)和純虛函數(shù)區(qū)別
觀點一:
類里聲明為虛函數(shù)的話,這個函數(shù)是實現(xiàn)的成黄,哪怕是空實現(xiàn)呐芥,它的作用就是為了能讓這個函數(shù)在它的子類里面可以被重載,這樣的話慨默,這樣編譯器就可以使用后期綁定來達到多態(tài)了
純虛函數(shù)只是一個接口贩耐,是個函數(shù)的聲明而已弧腥,它要留到子類里去實現(xiàn)厦取。
class A{
protected:
void foo();//普通類函數(shù)
virtual void foo1();//虛函數(shù)
virtual void foo2() = 0;//純虛函數(shù)
}
觀點二:
虛函數(shù)在子類里面也可以不重載的;但純虛必須在子類去實現(xiàn)管搪,這就像Java的接口一樣虾攻。通常我們把很多函數(shù)加上virtual,是一個好的習(xí)慣更鲁,雖然犧牲了一些性能霎箍,但是增加了面向?qū)ο蟮亩鄳B(tài)性,因為你很難預(yù)料到父類里面的這個函數(shù)不在子類里面不去修改它的實現(xiàn)
觀點三:
虛函數(shù)的類用于“實作繼承”澡为,繼承接口的同時也繼承了父類的實現(xiàn)漂坏。當(dāng)然我們也可以完成自己的實現(xiàn)。純虛函數(shù)的類用于“介面繼承”媒至,主要用于通信協(xié)議方面顶别。關(guān)注的是接口的統(tǒng)一性,實現(xiàn)由子類完成拒啰。一般來說驯绎,介面類中只有純虛函數(shù)的。
觀點四:
帶純虛函數(shù)的類叫虛基類谋旦,這種基類不能直接生成對象剩失,而只有被繼承屈尼,并重寫其虛函數(shù)后,才能使用拴孤。這樣的類也叫抽象類脾歧。
虛函數(shù)是為了繼承接口和默認行為
純虛函數(shù)只是繼承接口,行為必須重新定義