c++中虛函數(shù)的實(shí)現(xiàn)詳解
前言
c++ 分為編譯時(shí)多態(tài)和運(yùn)行時(shí)多態(tài)。運(yùn)行時(shí)多態(tài)依賴于虛函數(shù),大部分人或許聽說(shuō)過(guò)虛函數(shù)是由虛函數(shù)表+虛函數(shù)指針實(shí)現(xiàn)的,但,真的是這樣嗎?雖然 c++ 規(guī)范有著復(fù)雜的語(yǔ)言細(xì)節(jié),但底層實(shí)現(xiàn)機(jī)制卻任由編譯器廠商想象。(沒準(zhǔn)某種特殊的處理器電路結(jié)構(gòu)原生支持虛函數(shù),沒準(zhǔn)這個(gè)處理器壓根不是馮紐曼型,或者將來(lái)廠商發(fā)明了比虛函數(shù)表更有效率的數(shù)據(jù)結(jié)構(gòu)。)
虛函數(shù)表
封裝把實(shí)例的數(shù)據(jù)和操作結(jié)合在了一起,但實(shí)例本身只有數(shù)據(jù),沒有函數(shù),同一個(gè)類的函數(shù)是共享的。我們通過(guò)一個(gè)例子來(lái)間接證明這一點(diǎn)
class Base1 { public: int a; void func() { cout << "heel" << endl; } }; Base1 b1; cout << sizeof(b1) << endl;
打印
4
如果類中有虛函數(shù),則會(huì)在對(duì)象中加入一個(gè)虛函數(shù)指針,該指針指向一個(gè)虛函數(shù)表,表中是各個(gè)虛函數(shù)的地址。
+--------+ +---------+ | pvtbl |------>| vfunc1 | +--------+ +---------+ | data1 | | vfunc2 | +--------+ +---------+ | ... | | ... |
當(dāng)子類繼承父類時(shí),會(huì)依次覆蓋虛函數(shù)表中的各個(gè)項(xiàng),如果子類沒有重寫某項(xiàng),那該項(xiàng)就保留。當(dāng)實(shí)例化對(duì)象后,虛函數(shù)指針就作為一個(gè)隱藏?cái)?shù)據(jù)存在于實(shí)例中。如果通過(guò)父類指針調(diào)用普通成員函數(shù),由于普通函數(shù)和類型綁定在一起,所以仍會(huì)調(diào)用父類成員函數(shù);如果通過(guò)父類指針調(diào)用虛函數(shù),則會(huì)通過(guò)對(duì)象的虛指針找到虛函數(shù)表(即子類的虛函數(shù)表),定位虛函數(shù)項(xiàng),實(shí)現(xiàn)多態(tài)。
原理是不是很簡(jiǎn)單?c++ 就是通過(guò)這種看似原始的方式實(shí)現(xiàn)高級(jí)抽象。以上是編譯器的通用做法,我手上的 Visual Studio 2013 編譯器就是這么做的,為了提高性能,VS 保證虛函數(shù)指針存在于對(duì)象實(shí)例中最前面位置(歷史上也有編譯器不這么做,好像是 Borland 的?)。
Visual Studio 2013 中的實(shí)現(xiàn)
來(lái)一個(gè)例子(能這么寫是因?yàn)槲乙阎?Visual Studio 2013 編譯后對(duì)象的內(nèi)存布局)
#include <iostream> using namespace std; class Base { public: typedef void (*func)(); virtual void func1() { cout << "Base::func1" << endl; } virtual void func2() { cout << "Base::func2" << endl; } virtual void func3() { cout << "Base::func3" << endl; } }; class Derived: public Base { public: virtual void func1() { cout << "Derived::func1" << endl; } virtual void func3() { cout << "Derived::func3" << endl; } }; int main() { Base b, b1; int** pvirtualtable1 = (int**)&b; cout << "Base object vtbl address: " << pvirtualtable1[0] << endl; int** pvirtualtable11 = (int**)&b1; cout << "another Base object vtbl address: " << pvirtualtable11[0] << endl; cout << "function in virtual table" << endl; for (int i = 0; (Base::func)pvirtualtable1[0][i] != NULL; ++i) { auto p = (Base::func)pvirtualtable1[0][i]; p(); } cout << endl; Derived d; int** pvirtualtable2 = (int**)&d; cout << "Derived object vtbl address: " << pvirtualtable2[0] << endl; cout << "function in virtual table" << endl; for (int i = 0; (Base::func)pvirtualtable2[0][i] != NULL; ++i) { auto p = (Base::func)pvirtualtable2[0][i]; p(); } cout << endl; }
打印
Base object pvtbl address: 0029DA58 another Base object pvtbl address: 0029DA58 function address in virtual table Base::func1 Base::func2 Base::func3 Derived object pvtbl address: 0029DB20 function address in virtual table Derived::func1 Base::func2 Derived::func3
可以看到,同一類型不同實(shí)例的虛函數(shù)表是相同的,繼承之后,子類有了自己的虛函數(shù)表,表也有相應(yīng)的更新(Derived::func1, Derived::func3),表中未重寫的項(xiàng)還保留為原值(Base::func2)。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,如果有疑問大家可以留言交流。
上一篇:c++連接mysql5.6的出錯(cuò)問題總結(jié)
欄 目:C語(yǔ)言
下一篇:從txt中讀入數(shù)據(jù)到數(shù)組中(fscanf)的實(shí)現(xiàn)代碼
本文標(biāo)題:c++中虛函數(shù)的實(shí)現(xiàn)詳解
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/1904.html
您可能感興趣的文章
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用函數(shù)刪除字符
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)式函數(shù)庫(kù)
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)數(shù)怎么表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫分段 用c語(yǔ)言表示分段函數(shù)
- 04-02c語(yǔ)言編寫函數(shù)冒泡排序 c語(yǔ)言冒泡排序法函數(shù)
- 04-02c語(yǔ)言沒有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段函數(shù)
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎么打出三角函數(shù)的值
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求階乘


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫分段 用c語(yǔ)言表示分段
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)
- 04-02c語(yǔ)言編寫函數(shù)冒泡排序 c語(yǔ)言冒泡排
- 04-02c語(yǔ)言沒有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10delphi制作wav文件的方法
- 04-02jquery與jsp,用jquery
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?