C++虛函數(shù)
出處:Bingoes 發(fā)布于:2010-05-20 16:50:56
作者:陶寧,華清遠(yuǎn)見(jiàn)嵌入式學(xué)院上海中心講師。
虛函數(shù)是在類中被聲明為virtual的成員函數(shù),當(dāng)編譯器看到通過(guò)指針或引用調(diào)用此類函數(shù)時(shí),對(duì)其執(zhí)行晚綁定,即通過(guò)指針(或引用)指向的類的類型信息來(lái)決定該函數(shù)是哪個(gè)類的。通常此類指針或引用都聲明為基類的,它可以指向基類或派生類的對(duì)象。
多態(tài)指同一個(gè)方法根據(jù)其所屬的不同對(duì)象可以有不同的行為。
下面來(lái)看一個(gè)例子:

程序運(yùn)行輸出結(jié)果: 8, 12
為什么Base 對(duì)象的大小是8?jìng)€(gè)字節(jié)而不是4個(gè)字節(jié),為什么Derived 對(duì)象的大小是12個(gè)字節(jié)而不是8個(gè)字節(jié),多出來(lái)的4個(gè)字節(jié)做什么用呢?和多態(tài)的實(shí)現(xiàn)有什么關(guān)系?
每一個(gè)有虛函數(shù)的類(或有虛函數(shù)的類的派生類)都有一個(gè)虛函數(shù)表,該類的任何對(duì)象中都放著虛函數(shù)表的指針。虛函數(shù)表中列出了該類的虛函數(shù)地址。多出來(lái)的4個(gè)字節(jié)就是用來(lái)放虛函數(shù)表的地址的。
每當(dāng)創(chuàng)建一個(gè)包含虛函數(shù)的類或者從一個(gè)有虛函數(shù)的基類中派生一個(gè)類時(shí),編譯器就為這個(gè)類創(chuàng)建一個(gè)VTABLE,在這個(gè)表中,放置了在這個(gè)類中或它的基類中所有聲明為virtual的虛函數(shù)的地址。然后編譯器在這個(gè)類中放置VPTR指向相應(yīng)的VTABLE。VPTR必須在構(gòu)造函數(shù)中被初始化,在VPTR初始化之前,不能調(diào)用虛函數(shù)。所有的基類對(duì)象或者從基類派生出的對(duì)象的VPTR都在各自對(duì)象的相同位置。所有的VTABLE有相同的順序,不管何種類型的對(duì)象。
C++的函數(shù)調(diào)用與C一樣,都是從右向左進(jìn)棧的,其間,對(duì)象的首地址也即this指針的值被壓入棧,正因?yàn)檎{(diào)用每個(gè)成員函數(shù)時(shí)this都必須作為參數(shù)壓進(jìn)棧,所以成員函數(shù)知道它工作在哪個(gè)特殊對(duì)象上。這樣,我們總能看到,在成員函數(shù)調(diào)用之前壓棧的次數(shù)等于參數(shù)個(gè)數(shù)加一(除了static成員函數(shù),它沒(méi)有this)。
純虛函數(shù) :
抽象類就是在類的聲明前面加上virtual關(guān)鍵字,為了防止誤用抽象類,可以在抽象類中定義純虛函數(shù),例如virtual void x()=0;
這樣做,等于告訴編譯器在VTABLE中為函數(shù)保留一個(gè)間隔,但在這個(gè)特定間隔不放地址,只要有一個(gè)純虛函數(shù),則VTABLE就是不完全的,包含純虛函數(shù)的類稱為純抽象基類。
虛函數(shù)是C++中用于實(shí)現(xiàn)多態(tài)(polymorphism)的機(jī)制。理念就是通過(guò)基類訪問(wèn)派生類定義的函數(shù)。假設(shè)我們有下面的類層次:

那么,在使用的時(shí)候,我們可以:
A * a = new B();
a->foo(); // 在這里,a雖然是指向A的指針,但是被調(diào)用的函數(shù)(foo)卻是B的!
這個(gè)例子是虛函數(shù)的一個(gè)典型應(yīng)用,通過(guò)這個(gè)例子,也許你就對(duì)虛函數(shù)有了一些概念。它虛就虛在所謂“推遲聯(lián)編”或者“動(dòng)態(tài)聯(lián)編”上,一個(gè)類函數(shù)的調(diào)用并不是在編譯時(shí)刻被確定的,而是在運(yùn)行時(shí)刻被確定的。由于編寫代碼的時(shí)候并不能確定被調(diào)用的是基類的函數(shù)還是哪個(gè)派生類的函數(shù),所以被成為“虛”函數(shù)。
如下聲明表示一個(gè)函數(shù)為純虛函數(shù):
class A
{
public:
virtual void foo()=0; // =0標(biāo)志一個(gè)虛函數(shù)為純虛函數(shù)
};
一個(gè)函數(shù)聲明為純虛后,純虛函數(shù)的意思是:我是一個(gè)抽象類!不要把我實(shí)例化!純虛函數(shù)用來(lái)規(guī)范派生類的行為,實(shí)際上就是所謂的“接口”。它告訴使用者,我的派生類都會(huì)有這個(gè)函數(shù)。
“本文由華清遠(yuǎn)見(jiàn)https://www.embedu.org/index.htm提供”
版權(quán)與免責(zé)聲明
凡本網(wǎng)注明“出處:維庫(kù)電子市場(chǎng)網(wǎng)”的所有作品,版權(quán)均屬于維庫(kù)電子市場(chǎng)網(wǎng),轉(zhuǎn)載請(qǐng)必須注明維庫(kù)電子市場(chǎng)網(wǎng),http://www.hbjingang.com,違反者本網(wǎng)將追究相關(guān)法律責(zé)任。
本網(wǎng)轉(zhuǎn)載并注明自其它出處的作品,目的在于傳遞更多信息,并不代表本網(wǎng)贊同其觀點(diǎn)或證實(shí)其內(nèi)容的真實(shí)性,不承擔(dān)此類作品侵權(quán)行為的直接責(zé)任及連帶責(zé)任。其他媒體、網(wǎng)站或個(gè)人從本網(wǎng)轉(zhuǎn)載時(shí),必須保留本網(wǎng)注明的作品出處,并自負(fù)版權(quán)等法律責(zé)任。
如涉及作品內(nèi)容、版權(quán)等問(wèn)題,請(qǐng)?jiān)谧髌钒l(fā)表之日起一周內(nèi)與本網(wǎng)聯(lián)系,否則視為放棄相關(guān)權(quán)利。
- EDA技術(shù)工具鏈與全流程設(shè)計(jì)運(yùn)維指南2026/1/5 10:28:51
- PLC程序現(xiàn)場(chǎng)疑難問(wèn)題排查與深度優(yōu)化指南2025/12/24 14:36:36
- PLC程序現(xiàn)場(chǎng)調(diào)試與優(yōu)化實(shí)操指南2025/12/24 14:29:57
- 工業(yè)PLC模擬量信號(hào)采集:調(diào)理技術(shù)與抗干擾工程方案2025/12/15 14:39:08
- PLC設(shè)備如何選型2025/9/5 17:15:14
- 物聯(lián)網(wǎng)節(jié)點(diǎn)低功耗設(shè)計(jì):信號(hào)鏈中的濾波與功耗管理
- 同步整流中MOSFET的應(yīng)用要點(diǎn)
- 輸出短路對(duì)電源芯片的影響
- 連接器壽命評(píng)估與可靠性設(shè)計(jì)
- PCB電源完整性(PI)設(shè)計(jì)核心實(shí)操規(guī)范
- 多層PCB疊層設(shè)計(jì)核心實(shí)操規(guī)范
- 提高M(jìn)OSFET效率的電路優(yōu)化方法
- 電源管理IC在智能家居中的應(yīng)用
- 差分信號(hào)連接器設(shè)計(jì)要點(diǎn)
- PCB焊盤與過(guò)孔設(shè)計(jì)核心實(shí)操規(guī)范(含可焊性與可靠性保障)









