=========================定義=========================
將一個(gè)調(diào)用函數(shù)連接上正確的被調(diào)用函數(shù),這個(gè)過程就叫做函數(shù)的聯(lián)編,簡(jiǎn)稱聯(lián)編。在C++中,一共有兩種聯(lián)編的方式:
- 靜態(tài)聯(lián)編
#define:靜態(tài)聯(lián)編是指聯(lián)編工作出現(xiàn)在編譯連接階段。 #特點(diǎn):① 靜態(tài)聯(lián)編就是指被調(diào)用函數(shù)和執(zhí)行調(diào)用函數(shù)之間的關(guān)系以及它們?cè)趦?nèi)存中的地址在編譯的時(shí)候已經(jīng)確定好了,運(yùn)行時(shí)不會(huì)發(fā)生變化。 ② 由于對(duì)象不用對(duì)自身進(jìn)行跟蹤,因此速度浪費(fèi)比較小,但是靈活性較差。
- 動(dòng)態(tài)聯(lián)編
#define:動(dòng)態(tài)聯(lián)編是指在程序運(yùn)行的時(shí)候才進(jìn)行的聯(lián)編工作。 #特點(diǎn):① 由于編譯程序在編譯階段并不能確切知道將要調(diào)用的函數(shù),只有在程序執(zhí)行時(shí)才能確定將要調(diào)用的函數(shù)。要確切之道該調(diào)用的函數(shù),就必須要求聯(lián)編工作在程序運(yùn)行時(shí)才能進(jìn)行。 ② 雖然可以追蹤對(duì)象,靈活性較強(qiáng),但是速度浪費(fèi)嚴(yá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 | #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 :
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);
uniqueliu->print();
}
|
我們先來看一下這個(gè)程序的運(yùn)行結(jié)果:

上面這個(gè)結(jié)果很好理解吧,我就不解釋了。但是如果我們這里把程序33行的注釋去掉,把32行注釋起來。那么程序運(yùn)行的結(jié)果就會(huì)出現(xiàn)如下的情況:

這個(gè)就很奇怪了。我們的初衷是想用一個(gè)父類指針來指向一個(gè)子類對(duì)象。但是最后調(diào)用的print函數(shù)卻是父類對(duì)象的。這個(gè)原因就是因?yàn)槲覀儧]有在父類
中的print()前面加上關(guān)鍵字virtual的原因了。這是因?yàn)樵诤瘮?shù)print()函數(shù)前面加上了關(guān)鍵字virtual,就表示該函數(shù)是有多種形態(tài)
的,說白了,就是這個(gè)print()函數(shù)可以被很多對(duì)象所擁有,而且各自實(shí)現(xiàn)的功能是不一樣的,這樣一來,就是先了多態(tài)??偨Y(jié)一下,我們只要在基類的成員
函數(shù)前面加上virual,那么就算派生類的對(duì)象重新實(shí)現(xiàn)了同名函數(shù),編譯器就會(huì)自動(dòng)判斷是哪個(gè)對(duì)象調(diào)用了它,然后用該對(duì)象的同名函數(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 | #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 :
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();
father *uniqueliu= new son(22);
uniqueliu->print();
}
|
這樣,我們?cè)诔绦虻牡?0行上加上了virtual關(guān)鍵字。它就將基類的print()函數(shù)聲明成了虛函數(shù)。這樣一來,我們就可以調(diào)用子類的print()函數(shù)了,上圖發(fā)真相!

=========================virtual與聯(lián)編之間的關(guān)系=========================
相對(duì)于例1來講,在這里面調(diào)用的關(guān)系就是靜態(tài)聯(lián)編,因?yàn)樵诔绦虻木幾g階段就已經(jīng)把對(duì)象和它們所指向的函數(shù)緊緊地聯(lián)系在一起了,運(yùn)行的時(shí)候自然就不會(huì)有任何
改變。而對(duì)于例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ū)尋找與之對(duì)應(yīng)的對(duì)象,所以輸出才會(huì)于紅框所示,“son's age is
22”。
特別注意一點(diǎn)!!!只有在使用指針或者是引用的時(shí)候,才能夠?qū)崿F(xiàn)動(dòng)態(tài)聯(lián)編。
|