虛函數、純虛函數、重載函數的作用與區別

 虛函數爲了重載和多態的需要,在基類中是有定義的,即便定義是空,所以子類中可以重寫也可以不寫基類中的此函數!

純虛函數在基類中是沒有定義的,必須在子類中加以實現,很像java中的接口函數!

虛函數

引入原因:爲了方便使用多態特性,我們常常需要在基類中定義虛函數。

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(); //始終調用CManEat成員函數,不會調用 CChild 

p = &m_child;

p->Eat(); //如果子類實現(覆蓋)了該方法,則始終調用CChildEat函數

//不會調用CMan  Eat 方法;如果子類沒有實現該函數,則調用CManEat函數

p->Move(); //子類中沒有該成員函數,所以調用的是基類中的

純虛函數

引入原因:

1、同虛函數

2、在很多情況下,基類本身生成對象是不合情理的。例如,動物作爲一個基類可以派生出老虎、孔雀等子類,但動物本身生成對象明顯不合常理。

純虛函數就是基類只定義了函數體,沒有實現過程,定義方法如: virtual void Eat() = 0; 不要cpp中定義;純虛函數相當於接口,不能直接實例話,需要派生類來實現函數定義;

有的人可能在想,定義這些有什麼用啊 ,我覺得很有用,比如你想描述一些事物的屬性給別人,而自己不想去實現,就可以定義爲純虛函數。說的再透徹一些。比如蓋樓房,你是老闆,你給建築公司描述清楚你的樓房的特性,多少層,樓頂要有個花園什麼的建築公司就可以按照你的方法去實現了,如果你不說清楚這些,可能建築公司不太瞭解你需要樓房的特性。用純需函數就可以很好的分工合作了

虛函數和純虛函數區別

觀點一:

類裏聲明爲虛函數的話,這個函數是實現的,哪怕是空實現,它的作用就是爲了能讓這個函數在它的子類裏面可以被重載,這樣的話,這樣編譯器就可以使用後期綁定來達到多態了

純虛函數只是一個接口,是個函數的聲明而已,它要留到子類裏去實現。

class A{

protected:

void foo();//普通類函數

virtual void foo1();//虛函數

virtual void foo2() = 0;//純虛函數

}

觀點二:

虛函數在子類裏面也可以不重載的;但純虛必須在子類去實現,這就像Java的接口一樣。通常我們把很多函數加上virtual,是一個好的習慣,雖然犧牲了一些性能,但是增加了面向對象的多態性,因爲你很難預料到父類裏面的這個函數不在子類裏面不去修改它的實現

觀點三:

虛函數的類用於實作繼承,繼承接口的同時也繼承了父類的實現。當然我們也可以完成自己的實現。純虛函數的類用於介面繼承,主要用於通信協議方面。關注的是接口的統一性,實現由子類完成。一般來說,介面類中只有純虛函數的。

觀點四:

帶純虛函數的類叫虛基類,這種基類不能直接生成對象,而只有被繼承,並重寫其虛函數後,才能使用。這樣的類也叫抽象類。

虛函數是爲了繼承接口和默認行爲

純虛函數只是繼承接口,行爲必須重新定義


虛函數與重載函數的區別 

                
                
    重載函數在類型和參數數量上一定不相同,而重定義的虛函數則要求參數的類型和個數、函數返回類型相同; 
    虛函數必須是類的成員函數,重載的函數則不一定是這樣; 
    構造函數可以重載,但不能是虛函數,析構函數可以是虛函數。 
                        
                        

例1:

                               
 #include  < iostream.h > 
 class  CBase
 {
public:
    
virtual int func(unsigned char ch) {return --ch;}
}
 ;
 class  CDerive :  public  CBase
 {
    
int func(char ch) {return ++ch;}   //此爲函數重載
}
 ;
 void  main()
 {    CBase *p=new CDerive;
    
int n=p->func(40);        //調用基類的 func()
    cout<<" the result is : "<<n<<endl;
}
                     
運行結果:
       the result is : 39
                       
                  

例2:

                 
 #include  < iostream.h > 
 class  CBase
 {
public:
    
virtual int func(unsigned char ch) {return --ch;}
}
 ;
 class  CDerive :  public  CBase
 {
    
int func(unsigned char ch) {return ++ch;}      //此爲虛函數
}
 ;
 void  main()
 {    CBase *p=new CDerive;
    
int n=p->func(40);        //調用派生類的 func()
    cout<<" the result is : "<<n<<endl;
}
                
               
運行結果:
        the result is : 41

原文鏈接:http://blog.csdn.net/livelylittlefish/article/details/2171515
標籤: <無>
分享到 
0
 0 |  0


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章