0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

深度解析C++中的虛函數(shù)

jf_78858299 ? 來源:阿Q正磚 ? 作者:阿Q正磚 ? 2023-02-15 11:14 ? 次閱讀

虛函數(shù)作為C++的重要特性,讓人又愛又怕,愛它功能強(qiáng)大,但又怕駕馭不好,讓它反咬一口,今天我們用CPU的角度,撕掉語法的偽裝,重新認(rèn)識一下虛函數(shù)。

虛函數(shù)是C++實現(xiàn)面向?qū)ο笤O(shè)計及多態(tài)特性的重要手段。沒有虛函數(shù),C++和C的區(qū)別就不大,都需要借助大量的“函數(shù)指針”,進(jìn)行面向?qū)ο蟮某绦蛟O(shè)計(特別是功能擴(kuò)展方面)。

有了虛函數(shù)的存在,函數(shù)指針的使用率大大降低,代碼可讀性,代碼數(shù)量都能得到大幅度的改善。

最厲害的是,C++的虛函數(shù)實現(xiàn)機(jī)制,幾乎同時在空間、效率上獲得了最優(yōu)解。

學(xué)習(xí)C++,虛函數(shù)是一條必經(jīng)之路!

先來看兩段簡單代碼

圖片

讓我們先比較一下普通函數(shù)體與虛函數(shù)體有什么區(qū)別,顯然,兩個函數(shù)是完全一致的,虛函數(shù)跟普通函數(shù)一樣,都會夾帶一個隱藏參數(shù)this指針。所以,如你所見,虛函數(shù)在實現(xiàn)方面,跟普通函數(shù)沒有任何區(qū)別。

讓我們再看看調(diào)用它們的時候,會有什么不同

圖片

通過對比,大部分地方也是相同的,箭頭指的那兩條指令都是在輸入:隱藏參數(shù) this指針。唯一的區(qū)別是,調(diào)用普通函數(shù)時,call指令的目標(biāo)地址在編譯階段就確定了,也就是所謂的“靜態(tài)綁定”;但調(diào)用虛函數(shù)時,call指令只能根據(jù)rdx寄存器的值來確定函數(shù)的位置,也就是所謂的“動態(tài)綁定”。

再深入理解下這幾條指令

圖片

原來當(dāng)類A有虛函數(shù)的時候,類A就會偷偷生成一個隱藏成員變量,方便起見,我們給這個隱藏變量起一個名字:V(指針類型),V存放著虛函數(shù)表的地址,根據(jù)偏移,就可以得到要執(zhí)行的vtest_1 的地址,將其存在寄存器rdx里面,隨后一條:call rdx 指令,一個虛函數(shù)的調(diào)用就完成了。如果說,類的成員函數(shù)會夾帶隱藏參數(shù) this指針,還能接受的話,那么,我說類還會夾帶隱藏變量V,你能接受嗎?如果真的存在隱藏變量V,在哪里給V初始化呢?答案是在A的構(gòu)造函數(shù)中,把V初始化成類A的虛函數(shù)表地址,如下:

圖片

盡管我沒有寫構(gòu)造函數(shù),編譯器還是會給我 生成一個默認(rèn)的構(gòu)造函數(shù) ,它一定、必須要幫我完成隱藏變量V的初始化。

當(dāng)然,A有派生類B的話

圖片

那么隱藏變量V會在B的構(gòu)造函數(shù)中被初始化為B的虛函數(shù)表地址,從而保證A、B的虛函數(shù)相互獨立,井水不犯河水,但考慮到派生類B的構(gòu)造函數(shù),還會調(diào)用基類A的構(gòu)造函數(shù)。因此,變量V一會兒會被初始化成類A的虛函數(shù)表,一會又會被初始化成類B的虛函數(shù)表,為了避免暈頭,往往會禁止在構(gòu)造函數(shù)里面調(diào)用虛函數(shù)。

小結(jié)一下:

1、虛函數(shù)在函數(shù)體的實現(xiàn)方面跟普通函數(shù)沒有任何區(qū)別。

2、虛函數(shù)的調(diào)用需要借助類對象的隱藏變量 V(vptr)來完成,隱藏變量V(vptr)會在構(gòu)造函數(shù)中被初始化成虛函數(shù)表的內(nèi)存地址。

3、調(diào)用任何虛函數(shù)的套路都是一樣的,唯一的區(qū)別是要根據(jù)它們在虛函數(shù)表的位置設(shè)置正確的偏移量。

大家可以看看調(diào)用vtest_1()和調(diào)用vtest_2()的唯一區(qū)別是什么?

圖片

不得不佩服虛函數(shù)的實現(xiàn)方法,幾乎同時在效率的空間上得到了最優(yōu)解,因為虛函數(shù)的出現(xiàn),函數(shù)指針的使用率大大降低,如果你還是被函數(shù)指針困擾的時候,或許可以考慮一下虛函數(shù)。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • cpu
    cpu
    +關(guān)注

    關(guān)注

    68

    文章

    10863

    瀏覽量

    211782
  • C++
    C++
    +關(guān)注

    關(guān)注

    22

    文章

    2108

    瀏覽量

    73653
  • 虛函數(shù)
    +關(guān)注

    關(guān)注

    0

    文章

    8

    瀏覽量

    1700
收藏 人收藏

    評論

    相關(guān)推薦

    C++函數(shù)virtual詳解

    C++ 函數(shù)用于解決動態(tài)多態(tài)問題,函數(shù)的作用是允許在派生類
    發(fā)表于 09-07 11:27 ?2886次閱讀

    C++函數(shù)操作指南(含代碼示例)

    本期博客,我們來介紹C++函數(shù),并給出一些實際操作的建議。
    發(fā)表于 02-12 10:12 ?686次閱讀

    C++標(biāo)準(zhǔn)編程:函數(shù)與內(nèi)聯(lián)

      曾經(jīng)在討論C++的時候,經(jīng)常會問到:“函數(shù)能被聲明為內(nèi)聯(lián)嗎?”現(xiàn)在,我們幾乎聽不到這個問題了?,F(xiàn)在聽到的是:“你不應(yīng)該使print成為內(nèi)聯(lián)的。聲明一個
    發(fā)表于 05-03 11:53

    關(guān)于C++函數(shù)重載機(jī)制

    ,而且同類型的同名函數(shù)能夠更好地發(fā)揮多種功能.宏觀體現(xiàn)就是使用一個函數(shù)名字可以完成各種同類型但是不同細(xì)節(jié)的函數(shù)調(diào)用(例如,參數(shù)的類型不同,或者僅僅是多了一個控制量參數(shù)......).所以C++
    發(fā)表于 10-01 17:18

    什么是C++函數(shù)? 應(yīng)該怎么定義? 用途是什么?

    什么是C++函數(shù)? 應(yīng)該怎么定義? 主要用途是什么?
    發(fā)表于 11-08 06:58

    C++如何處理內(nèi)聯(lián)函數(shù)

    當(dāng)一個函數(shù)是內(nèi)聯(lián)和函數(shù)時,會發(fā)生代碼替換或使用表調(diào)用嗎? 為了弄 清楚內(nèi)聯(lián)和函數(shù),讓我們將
    發(fā)表于 11-29 11:59 ?28次下載

    C++程序設(shè)計教程之多態(tài)性與函數(shù)的詳細(xì)資料說明

    本文檔詳細(xì)介紹的是C++程序設(shè)計教程之多態(tài)性與函數(shù)的詳細(xì)資料說明主要資料包括了:1 多態(tài)性的概念,2 一個典型的例子,3 函數(shù),4 純
    發(fā)表于 03-14 16:39 ?5次下載
    <b class='flag-5'>C++</b>程序設(shè)計教程之多態(tài)性與<b class='flag-5'>虛</b><b class='flag-5'>函數(shù)</b>的詳細(xì)資料說明

    如何深度解析C++拷貝構(gòu)造函數(shù)詳細(xì)資料說明

    本文檔的主要內(nèi)容詳細(xì)介紹的是如何深度解析C++拷貝構(gòu)造函數(shù)詳細(xì)資料說明。
    發(fā)表于 07-05 17:41 ?0次下載
    如何<b class='flag-5'>深度</b><b class='flag-5'>解析</b><b class='flag-5'>C++</b>拷貝構(gòu)造<b class='flag-5'>函數(shù)</b>詳細(xì)資料說明

    如何在中斷C函數(shù)調(diào)用C++

    之前,我們在單片機(jī)程序開發(fā)時都會面對中斷函數(shù)。眾所周知的,這個中斷函數(shù)肯定是要用C函數(shù)來定義的。我在用C++進(jìn)行程序開發(fā)的時候就發(fā)現(xiàn)了一個需
    發(fā)表于 05-09 18:17 ?0次下載
    如何在中斷<b class='flag-5'>C</b><b class='flag-5'>函數(shù)</b><b class='flag-5'>中</b>調(diào)用<b class='flag-5'>C++</b>

    圖文詳解:C++表的剖析

    圖文詳解:C++表的剖析
    的頭像 發(fā)表于 06-29 14:23 ?2542次閱讀
    圖文詳解:<b class='flag-5'>C++</b><b class='flag-5'>虛</b>表的剖析

    EE-128:C++的DSP:從C++調(diào)用匯編類成員函數(shù)

    EE-128:C++的DSP:從C++調(diào)用匯編類成員函數(shù)
    發(fā)表于 04-16 17:04 ?2次下載
    EE-128:<b class='flag-5'>C++</b><b class='flag-5'>中</b>的DSP:從<b class='flag-5'>C++</b>調(diào)用匯編類成員<b class='flag-5'>函數(shù)</b>

    C++語法的inline內(nèi)聯(lián)函數(shù)詳解

    上節(jié)我們分析了C++基礎(chǔ)語法的const,static以及 this 指針,那么這節(jié)內(nèi)容我們來看一下 inline 內(nèi)聯(lián)函數(shù)吧! inline 內(nèi)聯(lián)函數(shù) 特征 相當(dāng)于把內(nèi)聯(lián)函數(shù)里面的內(nèi)
    的頭像 發(fā)表于 09-09 09:33 ?3363次閱讀

    C++如何用函數(shù)實現(xiàn)多態(tài)

    01 — C++函數(shù)探索 C++是一門面向?qū)ο笳Z言,在C++里運行時多態(tài)是由
    的頭像 發(fā)表于 09-29 14:18 ?1699次閱讀

    一文詳解函數(shù)及其相關(guān)知識點

    本期是C++基礎(chǔ)語法分享的第七節(jié),今天給大家來分享一下: (1)析構(gòu)函數(shù); (2)純函數(shù); (3)
    的頭像 發(fā)表于 10-13 10:14 ?7936次閱讀

    函數(shù),C++開發(fā)者如何有效利用

    函數(shù)是基類聲明的成員函數(shù),且使用者期望在派生類中將其重新定義。那么,在 C++ ,什么是
    的頭像 發(fā)表于 02-11 09:39 ?945次閱讀