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

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

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

Android編程精選 ? 來源:編程學習總站 ? 作者:寫代碼的牛頓 ? 2021-09-29 14:18 ? 次閱讀

01

C++虛函數(shù)探索

C++是一門面向對象語言,在C++里運行時多態(tài)是由虛函數(shù)和純虛函數(shù)實現(xiàn)的,現(xiàn)在我們看下在C++中如何用虛函數(shù)實現(xiàn)多態(tài)。先來看一段代碼。

// virtual_function.cpp : 此文件包含 “main” 函數(shù)。程序執(zhí)行將在此處開始并結束。 // #include 《iostream》 class Base { public: Base()

{ std::cout 《《 “Base::constructor run” 《《 std::endl; } virtual void fun1()

{ std::cout 《《 “Base::fun1 run” 《《 std::endl; } virtual void fun2() { std::cout 《《 “Base::fun2 run” 《《 std::endl; } virtual ~Base()

{ std::cout 《《 “Base::desconstructor run” 《《 std::endl; } }; class Derive : public Base { public: Derive() { std::cout 《《 “Derive::constructor run” 《《 std::endl; } void fun1() { std::cout 《《 “Derive::fun1 run” 《《 std::endl; } void fun3()

{ std::cout 《《 “Derive::fun3 run” 《《 std::endl; } ~Derive() { std::cout 《《 “Derive::desconstructor run” 《《 std::endl; } }; int main() { Derive* d = new Derive(); d-》fun1(); d-》fun2(); d-》fun3(); delete d; }

這段代碼編譯運行后輸出了:

Base::constructor run Derive::constructor run Derive::fun1 run Base::fun2 run Derive::fun3 run Derive::desconstructor run Base::desconstructor run

這段代碼里基類Base定義了虛函數(shù)fun1和fun2,派生類Derive有成員函數(shù)fun1和fun3,其中派生類覆蓋了繼承而來的基類虛函數(shù)fun1。在主函數(shù)里創(chuàng)建Derive類型對象指針d指向Derive類型對象。由于派生類Derive成員函數(shù)fun1覆蓋了基類Base成員函數(shù)fun1,因此通過d調用fun1實際調用的是派生類Derive類的成員函數(shù)fun1,而繼承而來的成員函數(shù)fun2沒有被覆蓋,則通過指針d調用fun2實際調用的是基類成員函數(shù)fun2。這里好像讓看不出虛函數(shù)有什么作用,那么我們將主函數(shù)修改如下:

int main() { Base* b = new Derive(); b-》fun1(); b-》fun2(); delete b; }

在程序里我們將創(chuàng)建一個基類指針b并指向的是派生類,并且調用delete釋放內存時使用的是基類指針b。編譯運行輸出結果如下:

Base::constructor run Derive::constructor run Derive::fun1 run Base::fun2 run Derive::desconstructor run Base::desconstructor run

通過基類指針b調用fun1函數(shù),實際調用的是派生類Derive的成員函數(shù)fun1,由于在派生類Derive中沒有定義成員函數(shù)fun2,因此通過基類指針b調用fun2函數(shù),實際調用的依舊是基類Base的成員函數(shù)fun2。代碼里雖然我們沒有對派生類的成員函數(shù)fun1加virtual,實際上派生類的成員函數(shù)fun1是虛函數(shù)。但是對于大多數(shù)C++初學者會有2個疑問的地方。1、通過基類指針b調用fun1函數(shù),實際調用的是派生類的成員函數(shù)fun1。2、通過delete釋放內存使用的是基類指針b,會調用派生類析構函數(shù)和基類析構函數(shù),成功釋放內存,不會存在內存泄露問題。

帶有虛函數(shù)的類稱為虛基類,子類繼承虛基類。在C++中虛基類有一個虛函數(shù)表指針保存虛函數(shù)表地址,而虛函數(shù)表保存函數(shù)地址,虛函數(shù)表并不在虛基類里,但是虛函數(shù)表指針在虛基類里,子類繼承虛基類,子類也就有了虛函數(shù)表指針。那么C++是如何通過虛函數(shù)表和虛函數(shù)表指針實現(xiàn)多態(tài)呢?打開VS2019,并用管理員身份運行“2019開發(fā)人員命令提示符”工具,如下圖所示:

輸入:cl /d1 reportSingleClassLayoutXXX [filename],XXX表示類名,[filename]表示類所在的.cpp文件路徑。這里我輸入源文件的派生類名和源文件路徑,回車輸出如下:

從輸出可以看出派生類從基類繼承了虛函數(shù)表指針vfptr,且占用字節(jié)數(shù)大小是4字節(jié),剛好就是一個指針占用字節(jié)數(shù)。虛函數(shù)表vftable里保存了派生類成員函數(shù)fun1,基類成員函數(shù)fun2的地址,由于派生類成員函數(shù)fun3不是虛函數(shù),因此虛函數(shù)表里沒有fun3的地址??吹竭@里我們就明白了,通過基類指針b調用fun1的過程:通過虛函數(shù)表指針vfptr找到虛函數(shù)表vftable,再通過虛函數(shù)表找到派生類成員函數(shù)fun1的地址,調用派生類成員函數(shù)fun1。而通過基類指針b調用fun2的過程則是:通過虛函數(shù)表指針vfptr找到虛函數(shù)表vftable,再通過虛函數(shù)表找到基類成員函數(shù)fun2的地址,調用基類成員函數(shù)fun2??吹竭@里,第一個疑問已經解開了,關鍵在于虛函數(shù)表指針和虛函數(shù)表。

在C++中有虛函數(shù)的類,其析構函數(shù)默認是虛析構函數(shù),只要是虛函數(shù)就會在虛函數(shù)表里有相應的函數(shù)地址,因此派生類里的虛函數(shù)表指針vfptr指向的虛函數(shù)表vftable必然保存著派生類析構函數(shù)的地址,類的析構過程:從繼承鏈的最底端到最頂端依次調用析構函數(shù),因此delete b調用過程:通過虛函數(shù)表指針vfptr找到虛函數(shù)表vftable,再通過虛函數(shù)表找到派生類析構函數(shù)地址,調用析構函數(shù)。

責任編輯:haq

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

    關注

    3

    文章

    4277

    瀏覽量

    62323
  • C++
    C++
    +關注

    關注

    21

    文章

    2100

    瀏覽量

    73453

原文標題:C++虛函數(shù)詳解

文章出處:【微信號:AndroidPush,微信公眾號:Android編程精選】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    使用C語言實現(xiàn)函數(shù)模板

      用C語言能不能實現(xiàn)一個通用的函數(shù),既能完成整數(shù)的相加,又能完成浮點數(shù)的相加?
    的頭像 發(fā)表于 11-09 11:38 ?83次閱讀

    C語言和C++結構體的區(qū)別

    同樣是結構體,看看在C語言和C++中有什么區(qū)別?
    的頭像 發(fā)表于 10-30 15:11 ?101次閱讀

    ostreamc++的用法

    ostream 是 C++ 標準庫中一個非常重要的類,它位于 頭文件(實際上,更常見的是通過包含 頭文件來間接包含 ,因為 包含了 和 )。 ostream 類及其派生類(如 std::cout
    的頭像 發(fā)表于 09-20 15:11 ?416次閱讀

    ModusToolbox 3.2c代碼包含c++代碼的正確步驟是什么?

    文件,但要在 main.c #include 它們時 會導致構建失敗。 將 main.c 重命名為 main.cpp 會導致標準 XMC 庫函數(shù)(如 XMC_GPIO_SetMo
    發(fā)表于 07-23 08:21

    C++實現(xiàn)類似instanceof的方法

    函數(shù),可實際上C++沒有。但是別著急,其實C++中有兩種簡單的方法可以實現(xiàn)類似Java的in
    的頭像 發(fā)表于 07-18 10:16 ?499次閱讀
    <b class='flag-5'>C++</b><b class='flag-5'>中</b><b class='flag-5'>實現(xiàn)</b>類似instanceof的方法

    FX2 CY7C68013A如何在C++環(huán)境中使用LoadEEPROM函數(shù)

    我使用的是 FX2 CY7C68013A 芯片。 我知道 CyUSB.NET 庫中有我需要的 LoadEEPROM 函數(shù)。 請問如何在 C++ 環(huán)境而不是 C#/CLR 環(huán)境中使用該
    發(fā)表于 05-31 06:59

    C/C++兩種宏實現(xiàn)方式

    #ifndef的方式受C/C++語言標準支持。它不僅可以保證同一個文件不會被包含多次,也能保證內容完全相同的兩個文件(或者代碼片段)不會被不小心同時包含。
    的頭像 發(fā)表于 04-19 11:50 ?527次閱讀

    鴻蒙OS開發(fā)實例:【Native C++

    使用DevEco Studio創(chuàng)建一個Native C++應用。應用采用Native C++模板,實現(xiàn)使用NAPI調用C標準庫的功能。使用C
    的頭像 發(fā)表于 04-14 11:43 ?2449次閱讀
    鴻蒙OS開發(fā)實例:【Native <b class='flag-5'>C++</b>】

    使用 MISRA C++:2023? 避免基于范圍的 for 循環(huán)中的錯誤

    在前兩篇博客,我們?向您介紹了新的 MISRA C++ 標準?和?C++ 的歷史?。在這篇博客,我們將仔細研究以 C++
    的頭像 發(fā)表于 03-28 13:53 ?709次閱讀
    使用 MISRA <b class='flag-5'>C++</b>:2023? 避免基于范圍的 for 循環(huán)中的錯誤

    如何理解運放電路短和斷?

    模擬電路,短和斷是兩個重要的概念,它們通常與運放電路有關。這兩個術語描述了運放電路的一些重要現(xiàn)象,認識它們對于電子工程師和電路設計
    的頭像 發(fā)表于 01-26 08:20 ?1499次閱讀
    如何理解運放電路<b class='flag-5'>中</b>的<b class='flag-5'>虛</b>短和<b class='flag-5'>虛</b>斷?

    C++簡史:C++是如何開始的

    的 MISRA C++:2023 博客系列的第二部分。 在這篇博客,我們將深入探討 C++ 的歷史、編程語言多年來的發(fā)展歷程以及它的下一步發(fā)展方向。
    的頭像 發(fā)表于 01-11 09:00 ?513次閱讀
    <b class='flag-5'>C++</b>簡史:<b class='flag-5'>C++</b>是如何開始的

    基于C/C++面向對象的方式封裝socket通信類

    掌握了基于 TCP 的套接字通信流程之后,為了方便使用,提高編碼效率,可以對通信操作進行封裝,本著有淺入深的原則,先基于 C 語言進行面向過程的函數(shù)封裝,然后再基于 C++ 進行面向
    的頭像 發(fā)表于 12-26 09:57 ?1263次閱讀

    C語言和C++那些不同的地方

    ++11標準。根據(jù)不同的標準,它們的功能也會有所不同,但是越新的版本支持的編譯器越少,所以本文討論的時候使用的C語言標準是C89,C++標準是C
    的頭像 發(fā)表于 12-07 14:29 ?889次閱讀
    <b class='flag-5'>C</b>語言和<b class='flag-5'>C++</b><b class='flag-5'>中</b>那些不同的地方

    c++多行注釋快捷鍵

    C++,多行注釋(也稱為塊注釋)是一種用于注釋大段代碼或多個語句的方法。當你希望暫時禁用一些代碼或者解釋特定部分代碼的作用時,多行注釋是非常有用的。
    的頭像 發(fā)表于 11-22 10:24 ?7931次閱讀

    探索C++的編程習慣與編程要點

    C++的類可以分為帶指針數(shù)據(jù)成員與不帶指針數(shù)據(jù)成員兩類,complex就屬于不帶指針成員的類。而這里要說的字符串類String,一般的實現(xiàn)會帶有一個char *指針。帶指針數(shù)據(jù)成員的類,需要自己實現(xiàn)class三大件:拷貝構造
    的頭像 發(fā)表于 11-14 09:25 ?371次閱讀
    探索<b class='flag-5'>C++</b>的編程習慣與編程要點