C++中虛函數(shù)與純虛函數(shù)的用法
本文較為深入的分析了C++中虛函數(shù)與純虛函數(shù)的用法,對于學(xué)習(xí)和掌握面向?qū)ο蟪绦蛟O(shè)計(jì)來說是至關(guān)重要的。具體內(nèi)容如下:
首先,面向?qū)ο蟪绦蛟O(shè)計(jì)(object-oriented programming)的核心思想是數(shù)據(jù)抽象、繼承、動(dòng)態(tài)綁定。通過數(shù)據(jù)抽象,可以使類的接口與實(shí)現(xiàn)分離,使用繼承,可以更容易地定義與其他類相似但不完全相同的新類,使用動(dòng)態(tài)綁定,可以在一定程度上忽略相似類的區(qū)別,而以統(tǒng)一的方式使用它們的對象。
虛函數(shù)的作用是實(shí)現(xiàn)多態(tài)性(Polymorphism),多態(tài)性是將接口與實(shí)現(xiàn)進(jìn)行分離,采用共同的方法,但因個(gè)體差異而采用不同的策略。純虛函數(shù)則是一種特殊的虛函數(shù)。虛函數(shù)聯(lián)系到多態(tài),多態(tài)聯(lián)系到繼承。所以本文中都是在繼承層次上做文章。沒了繼承,什么都沒得談。
一、虛函數(shù)
1 . 定義
在C++中,基類必須將它的兩種成員函數(shù)區(qū)分開來:一種是基類希望其派生類進(jìn)行覆蓋的函數(shù);另一種是基類希望派生類直接繼承而不要改變的函數(shù)。對于前者,基類通過在函數(shù)之前加上virtual關(guān)鍵字將其定義為虛函數(shù)(virtual)。
class Base{ // 基類 public: virtual int func(int n) const; }; class Derive_Class : public Base{ public: int func(int n) const; // 默認(rèn)也為虛函數(shù) };
當(dāng)我們在派生類中覆蓋某個(gè)函數(shù)時(shí),可以在函數(shù)前加virtual關(guān)鍵字。然而這不是必須的,因?yàn)橐坏┠硞€(gè)函數(shù)被聲明成虛函數(shù),則所有派生類中它都是虛函數(shù)。任何構(gòu)造函數(shù)之外的非靜態(tài)函數(shù)都可以是虛函數(shù)。派生類經(jīng)常(但不總是)覆蓋它繼承的虛函數(shù),如果派生類沒有覆蓋其基類中某個(gè)虛函數(shù),則該虛函數(shù)的行為類似于其他的普通成員,派生類會(huì)直接繼承其在基類中的版本。
2 . 動(dòng)態(tài)綁定
當(dāng)我們使用基類的引用(或指針)調(diào)用一個(gè)虛函數(shù)時(shí)將發(fā)生動(dòng)態(tài)綁定(dynamic binding)。因?yàn)槲覀冎钡竭\(yùn)行時(shí)才能知道到底調(diào)用了哪個(gè)版本的虛函數(shù),可能是基類中的版本也可能是派生類中的版本,判斷的依據(jù)是引用(或指針)所綁定的對象的真實(shí)類型。與非虛函數(shù)在編譯時(shí)綁定不同,虛函數(shù)是在運(yùn)行時(shí)選擇函數(shù)的版本,所以動(dòng)態(tài)綁定也叫運(yùn)行時(shí)綁定(run-time binding)。
3 . 靜態(tài)類型與動(dòng)態(tài)類型
靜態(tài)類型指的是變量聲明時(shí)的類型或表達(dá)式生成的類型,它在編譯時(shí)總是已知的;動(dòng)態(tài)類型指的是變量或表達(dá)式表示的內(nèi)存中的對象的類型,它直到運(yùn)行時(shí)才可知。當(dāng)且僅當(dāng)通過基類的指針或引用調(diào)用虛函數(shù)時(shí),才會(huì)在運(yùn)行時(shí)解析該調(diào)用,也只有在這種情況下對象的動(dòng)態(tài)類型才有可能與靜態(tài)類型不同。如果表達(dá)式既不是引用也不是指針,則它的動(dòng)態(tài)類型永遠(yuǎn)與靜態(tài)類型一致。
4 . final和override
派生類中如果定義了一個(gè)函數(shù)與基類中虛函數(shù)同名但形參列表不同,編譯器會(huì)認(rèn)為這是派生類新定義的函數(shù)。如果我們的意圖本是覆蓋虛函數(shù),則這種錯(cuò)誤很難發(fā)現(xiàn)。通過在派生類中的虛函數(shù)最后加override關(guān)鍵字使得意圖更加清晰。如果我們使用override標(biāo)記了某個(gè)函數(shù),但該函數(shù)并沒有覆蓋已存在的虛函數(shù),編譯器將報(bào)錯(cuò)。
class Base{ // 基類 public: virtual int func(int a, int b) const; }; class Derive_Class : public Base{ public: int func(int a) const override; // 報(bào)錯(cuò),沒有覆蓋虛函數(shù) };
如果我們定義一個(gè)類,并不希望它被繼承?;蛘呦M硞€(gè)函數(shù)不被覆蓋,則可以把類或者函數(shù)指定為final,則之后任何嘗試?yán)^承該類或覆蓋該函數(shù)的操作將引發(fā)錯(cuò)誤。
class Base final { /* */ }; // 基類不能被繼承 class Derive_Class : public Base { /* */ }; // 報(bào)錯(cuò) void func(int) const final; // 不允許后續(xù)的其他類覆蓋func(int)
5 . 回避虛函數(shù)的機(jī)制
在某些情況下,我們希望對虛函數(shù)的調(diào)用不要進(jìn)行動(dòng)態(tài)綁定,而是強(qiáng)迫其執(zhí)行虛函數(shù)的某個(gè)特定版本??梢允褂米饔糜蜻\(yùn)算符實(shí)現(xiàn)這一目的。
// 強(qiáng)行調(diào)用基類中定義的函數(shù)版本而不管baseP的動(dòng)態(tài)類型是什么 int a = baseP->Base::func(42);
如果一個(gè)派生類虛函數(shù)需要調(diào)用它的基類版本,但是沒有使用作用域運(yùn)算符,則在運(yùn)行時(shí)該調(diào)用將被解析為對派生類版本自身的調(diào)用,從而導(dǎo)致無限遞歸。
二、純虛函數(shù)
1 . 定義
為了方便使用多態(tài)特性,我們常常需要在基類中定義虛函數(shù)。在許多情況下,在基類中不能對虛函數(shù)給出有意義的實(shí)現(xiàn)。為了讓虛函數(shù)在基類什么也不做,引進(jìn)了“純虛函數(shù)”的概念,使函數(shù)無須定義。我們通過在函數(shù)體的位置(即在聲明語句的分號之前)書寫=0就可以將一個(gè)虛函數(shù)說明為純虛函數(shù)(pure virtual)。其中,=0只能出現(xiàn)在類內(nèi)部的虛函數(shù)聲明語句處:
class Base{ // 抽象基類 public: virtual int func(int n) const =0; };
需要注意的是,我們也可以為純虛函數(shù)提供定義,不過函數(shù)體必須定義在類的外部。
2 . 抽象基類
含有(或者未經(jīng)覆蓋直接繼承)純虛函數(shù)的類叫抽象基類(abstract base class)。抽象基類負(fù)責(zé)定義接口,而后續(xù)的其他類可以覆蓋該接口。如果派生類中沒有重新定義純虛函數(shù),而只是繼承基類的純虛函數(shù),則這個(gè)派生類仍然還是一個(gè)抽象基類。因?yàn)槌橄蠡惡屑兲摵瘮?shù)(沒有定義),所以我們不能創(chuàng)建一個(gè)抽象基類的對象,但可以聲明指向抽象基類的指針或引用。
Base base; // 錯(cuò)誤,不能實(shí)例化抽象基類
總結(jié):
①.虛函數(shù)必須實(shí)現(xiàn),不實(shí)現(xiàn)編譯器會(huì)報(bào)錯(cuò)。
②.父類和子類都有各自的虛函數(shù)版本。由多態(tài)方式在運(yùn)行時(shí)動(dòng)態(tài)綁定。
③.通過作用域運(yùn)算符可以強(qiáng)行調(diào)用指定的虛函數(shù)版本。
④.純虛函數(shù)聲明如下:virtual void funtion()=0; 純虛函數(shù)無需定義。包含純虛函數(shù)的類是抽象基類,抽象基類不能創(chuàng)建對象,但可以聲明指向抽象基類的指針或引用。
⑤.派生類實(shí)現(xiàn)了純虛函數(shù)以后,該純虛函數(shù)在派生類中就變成了虛函數(shù),其子類可以再對該函數(shù)進(jìn)行覆蓋。
⑥.析構(gòu)函數(shù)通常應(yīng)該是虛函數(shù),這樣就能確保在析構(gòu)時(shí)調(diào)用正確的析構(gòu)函數(shù)版本。
上一篇:C++實(shí)現(xiàn)廣度優(yōu)先搜索實(shí)例
欄 目:C語言
下一篇:C++中new與delete、malloc與free應(yīng)用分析
本文標(biāo)題:C++中虛函數(shù)與純虛函數(shù)的用法
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/3493.html
您可能感興趣的文章
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用函數(shù)刪除字符
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)式函數(shù)庫
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言中對數(shù)函數(shù)的表達(dá)式 c語言中對數(shù)怎么表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段函數(shù)
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排序法函數(shù)
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段函數(shù)
- 04-02C語言中怎么打出三角函數(shù) c語言中怎么打出三角函數(shù)的值
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求階乘


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達(dá)式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 01-10delphi制作wav文件的方法
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 08-05織夢dedecms什么時(shí)候用欄目交叉功能?
- 04-02jquery與jsp,用jquery
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文