|
=========================定義=========================
將一個(gè)調(diào)用函數(shù)連接上正確的被調(diào)用函數(shù),這個(gè)過程就叫做函數(shù)的聯(liá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)系以及它們在內(nèi)存中的地址在編譯的時(shí)候已經(jīng)確定好了,運(yùn)行時(shí)不會(huì)發(fā)生變化。 ② 由于對象不用對自身進(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)行。 ② 雖然可以追蹤對象,靈活性較強(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è)子類對象。但是最后調(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)。總結(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 | #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();
}
|
這樣,我們在程序的第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)編。
|