溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

詳解C++中虛函數(shù)

發(fā)布時(shí)間:2020-07-18 16:39:50 來(lái)源:億速云 閱讀:190 作者:小豬 欄目:編程語(yǔ)言

小編這次要給大家分享的是詳解C++中虛函數(shù),文章內(nèi)容豐富,感興趣的小伙伴可以來(lái)了解一下,希望大家閱讀完這篇文章之后能夠有所收獲。

虛函數(shù)

基類中使用virtual關(guān)鍵字聲明的函數(shù),稱為虛函數(shù)。
虛函數(shù)的實(shí)現(xiàn),通過(guò)虛函數(shù)表來(lái)實(shí)現(xiàn)的。即V-table 這個(gè)表中有一個(gè)類,用于儲(chǔ)存虛函數(shù)的地址。解決其繼承,覆蓋的問(wèn)題,用于保證其真實(shí)反映的函數(shù)。這樣有虛函數(shù)的實(shí)例,將會(huì)儲(chǔ)存在這個(gè)實(shí)例的內(nèi)存中。即用父類的指針,操作子類的時(shí)候,通過(guò)虛函數(shù)表來(lái)實(shí)現(xiàn)找尋到父類。

定義下方的一個(gè)類

class Base{
public:
	virtual void f(){
		cout << "Base::f" << endl;
	}
	virtual void g(){
		cout << "Base::g" << endl;
	}
	virtual void h(){
		cout << "Base::h" << endl;
	}
}

然后通過(guò)實(shí)例化得到虛函數(shù)表

Base* b = new Base();

	//Fun pFun = NULL;

	cout << "虛函數(shù)表地址 " << (int*)(&(*b)) << endl;	// 強(qiáng)制轉(zhuǎn)換成為指針類型,然后輸出
	cout << "虛函數(shù)表 - 第一個(gè)函數(shù)地址" << (int*)*(int*)(&(*b)) << endl;	// 取虛函數(shù)的地址,然后獲得虛函數(shù)的頭指向的第一個(gè)存儲(chǔ)函數(shù)的內(nèi)存空間。然后獲取第一個(gè)函數(shù)的內(nèi)存地址

然后查看輸出的結(jié)果

虛函數(shù)表地址 0xb31268
虛函數(shù)表 - 第一個(gè)函數(shù)地址0x4c1490

根據(jù)地址可以看到,指針b指向創(chuàng)建的實(shí)例的地址,其首地址儲(chǔ)存著虛函數(shù)表的地址,然后再次通過(guò)指針訪問(wèn),得到虛函數(shù)表的第一個(gè)函數(shù)的指針的地址。

一般繼承,沒(méi)有虛函數(shù)覆蓋

詳解C++中虛函數(shù)

在上方的繼承中,子類沒(méi)有重載任何父類的函數(shù),那么在虛函數(shù)列表中,就代表著

詳解C++中虛函數(shù)

虛函數(shù)按照聲明的順序放入表中。
父類的虛函數(shù),在子類的虛函數(shù)的前面。

一般繼承,有虛函數(shù)覆蓋

詳解C++中虛函數(shù)

此時(shí)內(nèi)存中地址如下

詳解C++中虛函數(shù)

注意,最重要的一點(diǎn)是,地址上,覆蓋的f()函數(shù),被放置到了父類的f()函數(shù)上。于是就可以有下方的程序

	Base *b = new Derive();
	b->f();

使用一個(gè)類型為Base的指針b指向一個(gè)新建的實(shí)例Derive(),此時(shí)對(duì)于指針b指向的虛函數(shù)表中,此時(shí)f()的地址被Devieive函數(shù)的地址被覆蓋, 形成如上的虛函數(shù)表。

此時(shí)訪問(wèn)成功

多重繼承 無(wú)虛函數(shù)的覆蓋

假設(shè)有如上的內(nèi)容

詳解C++中虛函數(shù)

此時(shí)虛函數(shù)表

詳解C++中虛函數(shù)

每個(gè)父類都有自己的虛表,子類的成員函數(shù)被放到第一個(gè)父類的表中。即第一個(gè)父類的表是按照聲明的順序來(lái)判斷。

多重繼承 有虛函數(shù)覆蓋

詳解C++中虛函數(shù)

此時(shí)表如下

詳解C++中虛函數(shù)

此時(shí)父類的被替換了。

這樣就實(shí)現(xiàn)了多重繼承,代碼如下

	Derive d;
	Base1 *b1 = &d;
	Base2 *b2 = &d;
	Base3 *b3 = &d;
	b1->f();
	b2->f();
	b3->f();

此時(shí)代碼如上。

純虛函數(shù)

virtual int area() = 0;

實(shí)現(xiàn)一個(gè)純虛函數(shù),此時(shí)可以在派生類中更好的定義純虛函數(shù)。

看完這篇關(guān)于詳解C++中虛函數(shù)的文章,如果覺(jué)得文章內(nèi)容寫(xiě)得不錯(cuò)的話,可以把它分享出去給更多人看到。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI