欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

c++中虛函數(shù)的實(shí)現(xiàn)詳解

 更新時(shí)間:2016年12月22日 16:59:43   投稿:daisy  
眾所周知探索c++對象內(nèi)部的實(shí)現(xiàn)是一件非常有趣的事情,虛函數(shù)在c++中的實(shí)現(xiàn)機(jī)制就是用虛表和虛指針,但是具體是怎樣的呢?下面這篇文章就來給大家實(shí)際檢驗(yàn)一下 Visual Studio 2013 編譯器在無優(yōu)化條件下,虛函數(shù)的實(shí)現(xiàn)。有需要的朋友們可以參考借鑒,下面來一起看看吧。

前言

c++ 分為編譯時(shí)多態(tài)和運(yùn)行時(shí)多態(tài)。運(yùn)行時(shí)多態(tài)依賴于虛函數(shù),大部分人或許聽說過虛函數(shù)是由虛函數(shù)表+虛函數(shù)指針實(shí)現(xiàn)的,但,真的是這樣嗎?雖然 c++ 規(guī)范有著復(fù)雜的語言細(xì)節(jié),但底層實(shí)現(xiàn)機(jī)制卻任由編譯器廠商想象。(沒準(zhǔn)某種特殊的處理器電路結(jié)構(gòu)原生支持虛函數(shù),沒準(zhǔn)這個(gè)處理器壓根不是馮紐曼型,或者將來廠商發(fā)明了比虛函數(shù)表更有效率的數(shù)據(jù)結(jié)構(gòu)。)

虛函數(shù)表

封裝把實(shí)例的數(shù)據(jù)和操作結(jié)合在了一起,但實(shí)例本身只有數(shù)據(jù),沒有函數(shù),同一個(gè)類的函數(shù)是共享的。我們通過一個(gè)例子來間接證明這一點(diǎn)

class Base1
{
public:
 int a;
 void func() { cout << "heel" << endl; }
};

Base1 b1;
cout << sizeof(b1) << endl;

打印

4

如果類中有虛函數(shù),則會(huì)在對象中加入一個(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í)例化對象后,虛函數(shù)指針就作為一個(gè)隱藏?cái)?shù)據(jù)存在于實(shí)例中。如果通過父類指針調(diào)用普通成員函數(shù),由于普通函數(shù)和類型綁定在一起,所以仍會(huì)調(diào)用父類成員函數(shù);如果通過父類指針調(diào)用虛函數(shù),則會(huì)通過對象的虛指針找到虛函數(shù)表(即子類的虛函數(shù)表),定位虛函數(shù)項(xiàng),實(shí)現(xiàn)多態(tài)。

原理是不是很簡單?c++ 就是通過這種看似原始的方式實(shí)現(xiàn)高級抽象。以上是編譯器的通用做法,我手上的 Visual Studio 2013 編譯器就是這么做的,為了提高性能,VS 保證虛函數(shù)指針存在于對象實(shí)例中最前面位置(歷史上也有編譯器不這么做,好像是 Borland 的?)。

Visual Studio 2013 中的實(shí)現(xiàn)

來一個(gè)例子(能這么寫是因?yàn)槲乙阎?Visual Studio 2013 編譯后對象的內(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é)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,如果有疑問大家可以留言交流。

相關(guān)文章

  • Qt 數(shù)據(jù)庫QSqlDatabase使用示例

    Qt 數(shù)據(jù)庫QSqlDatabase使用示例

    本文主要介紹了Qt數(shù)據(jù)庫QSqlDatabase使用示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-12-12
  • C/C++中線程基本概念與創(chuàng)建詳解

    C/C++中線程基本概念與創(chuàng)建詳解

    線程是在進(jìn)程中產(chǎn)生的一個(gè)執(zhí)行單元,是CPU調(diào)度和分配的最小單元,其在同一個(gè)進(jìn)程中與其他線程并行運(yùn)行,他們可以共享進(jìn)程內(nèi)的資源。本文就和大家一起聊聊線程基本概念以及如何創(chuàng)建多線程,需要的可以參考一下
    2022-09-09
  • C語言與C++中關(guān)于字符串使用的比較

    C語言與C++中關(guān)于字符串使用的比較

    字符串是我們再熟悉不過的東西了,任何語言中字符串都是基礎(chǔ)都要經(jīng)常用到,那么在不同語言中字符串的用法一樣嗎?下面我們來看看C語言與C++中字符串使用的比較
    2022-05-05
  • C++ Boost Container庫示例詳細(xì)講解

    C++ Boost Container庫示例詳細(xì)講解

    Boost是為C++語言標(biāo)準(zhǔn)庫提供擴(kuò)展的一些C++程序庫的總稱。Boost庫是一個(gè)可移植、提供源代碼的C++庫,作為標(biāo)準(zhǔn)庫的后備,是C++標(biāo)準(zhǔn)化進(jìn)程的開發(fā)引擎之一,是為C++語言標(biāo)準(zhǔn)庫提供擴(kuò)展的一些C++程序庫的總稱
    2022-11-11
  • C++繼承詳細(xì)介紹

    C++繼承詳細(xì)介紹

    這篇文章主要介紹了C++繼承詳情,在我們進(jìn)行開發(fā)的時(shí)候,我們經(jīng)常會(huì)遇到抽象出來的類之間具有繼承關(guān)系。一個(gè)類繼承了另外一個(gè)類,被繼承的類成為基類或父類,繼承的類成為子類或派生類,下面文章的詳細(xì)內(nèi)容,需要的小伙伴可以參考一下
    2022-01-01
  • C++實(shí)現(xiàn)LeetCode(90.子集合之二)

    C++實(shí)現(xiàn)LeetCode(90.子集合之二)

    這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(90.子集合之二),本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-07-07
  • C++實(shí)現(xiàn)優(yōu)酷土豆去視頻廣告的方法

    C++實(shí)現(xiàn)優(yōu)酷土豆去視頻廣告的方法

    這篇文章主要介紹了C++實(shí)現(xiàn)優(yōu)酷土豆去視頻廣告的方法,實(shí)例分析了C++實(shí)現(xiàn)屏蔽功能的相關(guān)技巧,需要的朋友可以參考下
    2015-04-04
  • C++三色球問題描述與算法分析

    C++三色球問題描述與算法分析

    這篇文章主要介紹了C++三色球問題描述與算法分析,結(jié)合注釋形式詳細(xì)講述了三色球問題的描述與相應(yīng)的算法設(shè)計(jì)思路,并給出了相關(guān)的實(shí)現(xiàn)方法,需要的朋友可以參考下
    2016-05-05
  • 全面了解結(jié)構(gòu)體、聯(lián)合體和枚舉類型

    全面了解結(jié)構(gòu)體、聯(lián)合體和枚舉類型

    下面小編就為大家?guī)硪黄媪私饨Y(jié)構(gòu)體、聯(lián)合體和枚舉類型。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2016-07-07
  • C++回溯算法之深度優(yōu)先搜索詳細(xì)介紹

    C++回溯算法之深度優(yōu)先搜索詳細(xì)介紹

    回溯在迷宮搜索中使用很常見,就是這條路走不通,然后返回前一個(gè)路口,繼續(xù)下一條路。回溯算法說白了就是窮舉法,下面讓我們一起來看看回溯算法中深度優(yōu)先搜索吧
    2023-01-01

最新評論