=========================定義=========================
將一個(gè)調(diào)用函數(shù)連接上正確的被調(diào)用函數(shù),這個(gè)過程就叫做函數(shù)的聯(lián)編,簡稱聯(lián)編。在C++中,一共有兩種聯(lián)編的方式:
=========================virtual的作用=========================
首先,我們來看幾個(gè)例子:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 | //例1 用父類對象的指針指向子類對象 #include <iostream> using namespace std; class father { public : father(){}; father( int i){age=i;} void print() const {cout<< "father's age is " <<age<<endl;} protected : //protected聲明的成員可以被派生類訪問 int age; }; class son: public father { public : son(){}; son( int j){age=j;} void print() const {cout<< "son's age is " <<age<<endl;} }; int main() { cout<< "一般情況:\n" ; father dad(56); dad.print(); son boy(22); boy.print(); cout<< "指針情況:\n" ; father *liu= new father(56); liu->print(); son *uniqueliu= new son(22); //father *uniqueliu=new son(22); uniqueliu->print(); } |
我們先來看一下這個(gè)程序的運(yùn)行結(jié)果:
上面這個(gè)結(jié)果很好理解吧,我就不解釋了。但是如果我們這里把程序33行的注釋去掉,把32行注釋起來。那么程序運(yùn)行的結(jié)果就會(huì)出現(xiàn)如下的情況:
這個(gè)就很奇怪了。我們的初衷是想用一個(gè)父類指針來指向一個(gè)子類對象。但是最后調(diào)用的print函數(shù)卻是父類對象的。這個(gè)原因就是因?yàn)槲覀儧]有在父類中的print()前面加上關(guān)鍵字virtual的原因了。這是因?yàn)樵诤瘮?shù)print()函數(shù)前面加上了關(guān)鍵字virtual,就表示該函數(shù)是有多種形態(tài)的,說白了,就是這個(gè)print()函數(shù)可以被很多對象所擁有,而且各自實(shí)現(xiàn)的功能是不一樣的,這樣一來,就是先了多態(tài)??偨Y(jié)一下,我們只要在基類的成員函數(shù)前面加上virual,那么就算派生類的對象重新實(shí)現(xiàn)了同名函數(shù),編譯器就會(huì)自動(dòng)判斷是哪個(gè)對象調(diào)用了它,然后用該對象的同名函數(shù),而不會(huì)采用基類的函數(shù)了。程序如下所示:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 | //例2 用父類對象的指針指向子類對象+virtual #include <iostream> using namespace std; class father { public : father(){}; father( int i){age=i;} virtual void print() const {cout<< "father's age is " <<age<<endl;} protected : //protected聲明的成員可以被派生類訪問 int age; }; class son: public father { public : son(){}; son( int j){age=j;} void print() const {cout<< "son's age is " <<age<<endl;} }; int main() { cout<< "一般情況:\n" ; father dad(56); dad.print(); son boy(22); boy.print(); cout<< "指針情況:\n" ; father *liu= new father(56); liu->print(); //son *uniqueliu=new son(22); father *uniqueliu= new son(22); uniqueliu->print(); } |
這樣,我們在程序的第10行上加上了virtual關(guān)鍵字。它就將基類的print()函數(shù)聲明成了虛函數(shù)。這樣一來,我們就可以調(diào)用子類的print()函數(shù)了,上圖發(fā)真相!
=========================virtual與聯(lián)編之間的關(guān)系=========================
相對于例1來講,在這里面調(diào)用的關(guān)系就是靜態(tài)聯(lián)編,因?yàn)樵诔绦虻木幾g階段就已經(jīng)把對象和它們所指向的函數(shù)緊緊地聯(lián)系在一起了,運(yùn)行的時(shí)候自然就不會(huì)有任何改變。而對于例2來講,由于在父類的[rint()函數(shù)前面加上了關(guān)鍵字virtual,那么這個(gè)時(shí)候父類中的print()函數(shù)就變成了虛函數(shù),虛函數(shù)就可以實(shí)現(xiàn)運(yùn)行時(shí)的動(dòng)態(tài)聯(lián)編。這是因?yàn)関irtual會(huì)讓編譯器自動(dòng)地區(qū)尋找與之對應(yīng)的對象,所以輸出才會(huì)于紅框所示,“son's age is 22”。
特別注意一點(diǎn)?。?!只有在使用指針或者是引用的時(shí)候,才能夠?qū)崿F(xiàn)動(dòng)態(tài)聯(lián)編。