亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

詳解C++編程中的虛函數

2020-05-23 14:14:26
字體:
來源:轉載
供稿:網友

這篇文章主要介紹了詳解C++編程中的虛函數,包括在什么情況下應當聲明虛函數的相關講解,需要的朋友可以參考下

我們知道,在同一類中是不能定義兩個名字相同、參數個數和類型都相同的函數的,否則就是“重復定義”。但是在類的繼承層次結構中,在不同的層次中可以出現名字相同、參數個數和類型都相同而功能不同的函數。

人們提出這樣的設想,能否用同一個調用形式,既能調用派生類又能調用基類的同名函數。在程序中不是通過不同的對象名去調用不同派生層次中的同名函數,而是通過指針調用它們。例如,用同一個語句“pt->display( );”可以調用不同派生層次中的display函數,只需在調用前給指針變量 pt 賦以不同的值(使之指向不同的類對象)即可。

打個比方,你要去某一地方辦事,如果乘坐公交車,必須事先確定目的地,然后乘坐能夠到達目的地的公交車線路。如果改為乘出租車,就簡單多了,不必查行車路線,因為出租車什么地方都能去,只要在上車后臨時告訴司機要到哪里即可。如果想訪問多個目的地,只要在到達一個目的地后再告訴司機下一個目的地即可,顯然,“打的”要比乘公交車 方便。無論到什么地方去都可以乘同—輛出租車。這就是通過同一種形式能達到不同目的的例子。

C++中的虛函數就是用來解決這個問題的。虛函數的作用是允許在派生類中重新定義與基類同名的函數,并且可以通過基類指針或引用來訪問基類和派生類中的同名函數。

請分析下面這個例子。這個例子開始時沒有使用虛函數,然后再討論使用虛函數的情況。

[例] 基類與派生類中有同名函數。在下面的程序中Student是基類,Graduate是派生類,它們都有display這個同名的函數。

 

 
  1. #include <iostream> 
  2. #include <string> 
  3. using namespace std; 
  4. //聲明基類Student 
  5. class Student 
  6. public
  7. Student(int, string,float); //聲明構造函數 
  8. void display( );//聲明輸出函數 
  9. protected//受保護成員,派生類可以訪問 
  10. int num; 
  11. string name; 
  12. float score; 
  13. }; 
  14. //Student類成員函數的實現 
  15. Student::Student(int n, string nam,float s)//定義構造函數 
  16. num=n; 
  17. name=nam; 
  18. score=s; 
  19. void Student::display( )//定義輸出函數 
  20. cout<<"num:"<<num<<"/nname:"<<name<<"/nscore:"<<score<<"/n/n"
  21. //聲明公用派生類Graduate 
  22. class Graduate:public Student 
  23. public
  24. Graduate(int, string, floatfloat);//聲明構造函數 
  25. void display( );//聲明輸出函數 
  26. private:float pay; 
  27. }; 
  28. // Graduate類成員函數的實現 
  29. void Graduate::display( )//定義輸出函數 
  30. cout<<"num:"<<num<<"/nname:"<<name<<"/nscore:"<<score<<"/npay="<<pay<<endl; 
  31. Graduate::Graduate(int n, string nam,float s,float p):Student(n,nam,s),pay(p){} 
  32. //主函數 
  33. int main() 
  34. Student stud1(1001,"Li",87.5);//定義Student類對象stud1 
  35. Graduate grad1(2001,"Wang",98.5,563.5);//定義Graduate類對象grad1 
  36. Student *pt=&stud1;//定義指向基類對象的指針變量pt 
  37. pt->display( ); 
  38. pt=&grad1; 
  39. pt->display( ); 
  40. return 0; 

運行結果如下:

 

 
  1. num:1001(stud1的數據) 
  2. name:Li 
  3. score:87.5 
  4.  
  5. num:2001 (grad1中基類部分的數據) 
  6. name:wang 
  7. score:98.5 

假如想輸出grad1的全部數據成員,當然也可以采用這樣的方法:通過對象名調用display函數,如grad1.display(),或者定義一個指向Graduate類對象的指針變量ptr,然后使ptr指向gradl,再用ptr->display()調用。這當然是可以的,但是如果該基類有多個派生類,每個派生類又產生新的派生類,形成了同一基類的類族。每個派生類都有同名函數display,在程序中要調用同一類族中不同類的同名函數,就要定義多個指向各派生類的指針變量。這兩種辦法都不方便,它要求在調用不同派生類的同名函數時采用不同的調用方式,正如同前面所說的那樣,到不同的目的地要乘坐指定的不同的公交車,一一 對應,不能搞錯。如果能夠用同一種方式去調用同一類族中不同類的所有的同名函數,那就好了。

用虛函數就能順利地解決這個問題。下面對程序作一點修改,在Student類中聲明display函數時,在最左面加一個關鍵字virtual,即

 

 
  1. virtual void display( ); 

這樣就把Student類的display函數聲明為虛函數。程序其他部分都不改動。再編譯和運行程序,請注意分析運行結果:

 

 
  1. num:1001(stud1的數據) 
  2. name:Li 
  3. score:87.5 
  4.  
  5. num:2001 (grad1中基類部分的數據) 
  6. name:wang 
  7. score:98.5 
  8. pay=1200 (這一項以前是沒有的) 

看!這就是虛函數的奇妙作用?,F在用同一個指針變量(指向基類對象的指針變量),不但輸出了學生stud1的全部數據,而且還輸出了研究生grad1的全部數據,說明已調用了grad1的display函數。用同一種調用形式“pt->display()”,而且pt是同一個基類指針,可以調用同一類族中不同類的虛函數。這就是多態性,對同一消息,不同對象有 不同的響應方式。

說明:本來基類指針是用來指向基類對象的,如果用它指向派生類對象,則進行指針類型轉換,將派生類對象的指針先轉換為基類的指針,所以基類指針指向的是派生類對象中的基類部分。在程序修改前,是無法通過基類指針去調用派生類對象中的成員函數的。虛函數突破了這一限制,在派生類的基類部分中,派生類的虛函數取代了基類原來的虛函數,因此在使基類指針指向派生類對象后,調用虛函數時就調用了派生類的虛函數。 要注意的是,只有用virtual聲明了虛函數后才具有以上作用。如果不聲明為虛函數,企圖通過基類指針調用派生類的非虛函數是不行的。

虛函數的以上功能是很有實用意義的。在面向對象的程序設計中,經常會用到類的繼承,目的是保留基類的特性,以減少新類開發的時間。但是,從基類繼承來的某些成員函數不完全適應派生類的需要,例如在例中,基類的display函數只輸出基類的數據,而派生類的display函數需要輸出派生類的數據。過去我們曾經使派生類的輸出函數與基類的輸出函數不同名(如display和display1),但如果派生的層次多,就要起許多不同的函數名,很不方便。如果采用同名函數,又會發生同名覆蓋。

利用虛函數就很好地解決了這個問題。可以看到:當把基類的某個成員函數聲明為虛函數后,允許在其派生類中對該函數重新定義,賦予它新的功能,并且可以通過指向基類的指針指向同一類族中不同類的對象,從而調用其中的同名函數。由虛函數實現的動態多態性就是:同一類族中不同類的對象,對同一函數調用作出不同的響應。

虛函數的使用方法是:

在基類用virtual聲明成員函數為虛函數。

這樣就可以在派生類中重新定義此函數,為它賦予新的功能,并能方便地被調用。在類外定義虛函數時,不必再加virtual。

在派生類中重新定義此函數,要求函數名、函數類型、函數參數個數和類型全部與基類的虛函數相同,并根據派生類的需要重新定義函數體。

C++規定,當一個成員函數被聲明為虛函數后,其派生類中的同名函數都自動成為虛函數。因此在派生類重新聲明該虛函數時,可以加virtual,也可以不加,但習慣上一般在每一層聲明該函數時都加virtual,使程序更加清晰。如果在派生類中沒有對基類的虛函數重新定義,則派生類簡單地繼承其直接基類的虛函數。

定義一個指向基類對象的指針變量,并使它指向同一類族中需要調用該函數的對象。

通過該指針變量調用此虛函數,此時調用的就是指針變量指向的對象的同名函數。

通過虛函數與指向基類對象的指針變量的配合使用,就能方便地調用同一類族中不同類的同名函數,只要先用基類指針指向即可。如果指針不斷地指向同一類族中不同類的對象,就能不斷地調用這些對象中的同名函數。這就如同前面說的,不斷地告訴出租車司機要去的目的地,然后司機把你送到你要去的地方。

需要說明;有時在基類中定義的非虛函數會在派生類中被重新定義(如例中的area函數),如果用基類指針調用該成員函數,則系統會調用對象中基類部分的成員函數;如果用派生類指針調用該成員函數,則系統會調用派生類對象中的成員函數,這并不是多態性行為(使用的是不同類型的指針),沒有用到虛函數的功能。

以前介紹的函數重載處理的是同一層次上的同名函數問題,而虛函數處理的是不同派生層次上的同名函數問題,前者是橫向重載,后者可以理解為縱向重載。但與重載不同的是:同一類族的虛函數的首部是相同的,而函數重載時函數的首部是不同的(參數個數或類型不同)。

在什么情況下應當聲明虛函數

使用虛函數時,有兩點要注意:

只能用virtual聲明類的成員函數,使它成為虛函數,而不能將類外的普通函數聲明為虛函數。因為虛函數的作用是允許在派生類中對基類的虛函數重新定義。顯然,它只能用于類的繼承層次結構中。

一個成員函數被聲明為虛函數后,在同一類族中的類就不能再定義一個非virtual的但與該虛函數具有相同的參數(包括個數和類型)和函數返回值類型的同名函數。

根據什么考慮是否把一個成員函數聲明為虛函數呢?主要考慮以下幾點:

首先看成員函數所在的類是否會作為基類。然后看成員函數在類的繼承后有無可能被更改功能,如果希望更改其功能的,一般應該將它聲明為虛函數。

如果成員函數在類被繼承后功能不需修改,或派生類用不到該函數,則不要把它聲明為虛函數。不要僅僅考慮到要作為基類而把類中的所有成員函數都聲明為虛函數。

應考慮對成員函數的調用是通過對象名還是通過基類指針或引用去訪問,如果是通過基類指針或引用去訪問的,則應當聲明為虛函數。

有時,在定義虛函數時,并不定義其函數體,即函數體是空的。它的作用只是定義了一個虛函數名,具體功能留給派生類去添加。

需要說明的是:使用虛函數,系統要有一定的空間開銷。當一個類帶有虛函數時,編譯系統會為該類構造一個虛函數表(virtual function table,簡稱vtable),它是一個指針數組,存放每個虛函數的入口地址。系統在進行動態關聯時的時間開銷是很少的,因此,多態性是高效的。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美一级电影免费在线观看| 国产成人精品网站| 欧美激情亚洲视频| 久久人人爽亚洲精品天堂| 国产精品久久久久久久久久久新郎| 国产精品入口免费视频一| 亚洲欧美综合精品久久成人| 欧美日韩亚洲精品一区二区三区| 97久久国产精品| 久久久久久国产免费| 久久精品国亚洲| 国产99久久精品一区二区永久免费| 亚洲欧美综合精品久久成人| 国产成人精品视频在线| 91精品国产91| 国产精品亚洲美女av网站| 欧美在线视频a| 欧亚精品在线观看| 97免费中文视频在线观看| 欧美在线一区二区视频| 亚洲第一视频在线观看| www.99久久热国产日韩欧美.com| 日本欧美精品在线| 亚洲欧美日韩一区二区在线| 色偷偷888欧美精品久久久| 欧美又大粗又爽又黄大片视频| 91理论片午午论夜理片久久| 怡红院精品视频| 成人a在线视频| 欧美午夜性色大片在线观看| 国产经典一区二区| 国产亚洲精品久久久久久777| 精品一区二区三区电影| 亚洲肉体裸体xxxx137| 最近2019中文字幕mv免费看| 俺去亚洲欧洲欧美日韩| 精品久久香蕉国产线看观看gif| 亚洲男人天堂久| 中文字幕久久亚洲| 欧美日韩成人免费| 成人黄色在线播放| 成人性教育视频在线观看| 久久国产精品久久久久久| 日韩在线激情视频| 久久久久久久久亚洲| 国产主播在线一区| 欧美裸身视频免费观看| 欧美成人第一页| 精品电影在线观看| 日韩电影大全免费观看2023年上| 日韩欧美中文字幕在线观看| 日本国产一区二区三区| 亚洲欧美中文日韩在线v日本| 6080yy精品一区二区三区| 欧美成人国产va精品日本一级| 日本午夜人人精品| 欧美亚洲激情视频| 久久精品久久久久久| 一本一本久久a久久精品综合小说| 中文字幕日韩在线视频| 欧美成人精品三级在线观看| 亚洲国产成人一区| 久久国内精品一国内精品| 亚洲第一视频网站| 成人亚洲激情网| 九九综合九九综合| 亚洲精品福利视频| 高清欧美一区二区三区| 久久99青青精品免费观看| 91系列在线观看| 欧美视频在线免费| 国产精品久久电影观看| 国产视频一区在线| 97热在线精品视频在线观看| 日韩高清人体午夜| 国内自拍欧美激情| 亚洲第一精品夜夜躁人人爽| 欧美国产精品日韩| 在线a欧美视频| 久久久久久国产精品| 97福利一区二区| 91沈先生在线观看| 久久久极品av| 久久免费国产精品1| 亚洲aa中文字幕| 国产99久久久欧美黑人| 亚洲国产欧美精品| 精品亚洲一区二区三区在线观看| 91中文字幕一区| 秋霞av国产精品一区| 亚洲精品久久久一区二区三区| 黄色一区二区在线观看| 91精品在线播放| 精品一区二区三区四区| 国产一区二区欧美日韩| 国产精品扒开腿做| 亚洲免费一在线| 久久久久亚洲精品国产| 欧美另类69精品久久久久9999| 日韩av网站导航| 日本三级韩国三级久久| 最好看的2019的中文字幕视频| 91色琪琪电影亚洲精品久久| 国产午夜精品全部视频在线播放| 国产一区二区三区在线观看视频| 一区二区三区四区精品| 大量国产精品视频| 欧美激情伊人电影| 亚洲色图av在线| 国产日韩欧美日韩大片| 欧美日韩精品在线播放| 国产91精品不卡视频| 亚洲性日韩精品一区二区| 国产精品∨欧美精品v日韩精品| 91av在线免费观看视频| 奇米一区二区三区四区久久| 777国产偷窥盗摄精品视频| 成人黄色网免费| 国产精品久久久久久久久久久久久久| 亚洲天堂成人在线视频| 97婷婷涩涩精品一区| 亚洲va久久久噜噜噜久久天堂| 亚洲欧美在线免费| 日本亚洲欧美三级| 国产精品一区二区三| 91久久国产精品91久久性色| 亚洲一区二区在线播放| 日韩欧美一区视频| 亚洲另类xxxx| 亚洲国产精品电影| 成人在线精品视频| 欧美亚洲国产视频小说| 久久久久日韩精品久久久男男| 中文一区二区视频| 中文字幕亚洲精品| 亚洲综合中文字幕在线| 国产日本欧美视频| 欧美激情视频三区| 国产精品毛片a∨一区二区三区|国| 97婷婷大伊香蕉精品视频| 日韩在线观看免费全| 在线视频日本亚洲性| 成人激情视频免费在线| 精品久久久国产精品999| 成人高清视频观看www| 91av在线国产| 欧美激情按摩在线| 91精品国产高清自在线看超| 亚洲自拍另类欧美丝袜| 亚洲免费伊人电影在线观看av| 日韩电影中文字幕在线观看| 欧美日韩国产丝袜美女| 97人人爽人人喊人人模波多| 成人淫片在线看| 亚洲91av视频| 91精品久久久久久久久久久久久| 欧洲美女7788成人免费视频| 亚洲xxxxx电影| 亚洲国产欧美一区| 精品高清一区二区三区| 91精品国产综合久久香蕉最新版| 性欧美激情精品| 精品成人在线视频| 欧美精品电影在线|