国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - C/C++ - 聊一聊C++虛函數表的問題

聊一聊C++虛函數表的問題

2022-01-24 14:25碼農小明 C/C++

C++是面向對象的語言(與C語言主要區別),所以C++也擁有多態的特性。下面通過代碼看下C++虛函數表的問題,感興趣的朋友一起看看吧

之前只是看過C++虛函數表相關介紹,今天有空就來寫代碼研究一下。

面向對象的編程語言有3大特性:封裝、繼承和多態。C++是面向對象的語言(與C語言主要區別),所以C++也擁有多態的特性。

C++中多態分為兩種:靜態多態動態多態

靜態多態為編譯器在編譯期間就可以根據函數名和參數等信息確定調用某個函數。靜態多態主要體現為函數重載運算符重載

函數重載即類中定義多個同名成員函數,函數參數類型、參數個數和返回值不完全相同,編譯器編譯后這些同名函數的函數名會不一樣,也就是說編譯期間就確定了調用某個函數。C語言函數編譯后函數名就是原函數名,C++函數名為原函數名拼接函數參數等信息。

動態多態即運行時多態,在程序執行期間(非編譯期)判斷所引用對象的實際類型,根據其實際類型調用相應的方法。動態多態由虛函數來實現。
比如

class Base{};
class A: public Base{};
class A: public Base{};

Base *base = new A; // base靜態類型為Base*,動態類型為A*
base = new B; // base動態類型變為B*了

 

探索虛函數表結構

之前的文件提到過,一個類占用的空間,如果有虛函數就會占用8字節的空間來存放虛函數表的地址。
虛函數表內存空間 中依次存放著各個虛函數的指針,通過這個指針可以調用相關的虛函數。

下面通過代碼來驗證一下上面這個內存結構,定義一個Base類,中間有3個方法,f1/f2/f3。

class Base {
public:
  virtual void f1(){
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }

  virtual void f2(){
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }

  virtual void f3(){
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }
};

實例化這個類后的內存模型如下圖所示:
聊一聊C++虛函數表的問題

下面通過代碼來驗證這個內存模型。

int main() {
  typedef void(*Fun)();  // Fun為f1 f2 f3的函數類型

  std::cout << sizeof(Base)<< std::endl;  // 輸出 8

  Base b;
  printf("b ptr = %p\n", &b);  // b ptr = 0x7ffeee41ac30

  long v_table_addr_value = *(long*)&b; // 取&b指針 前8字節的值,即虛函數表地址值
  printf("vtable ptr = 0x%lx\n", v_table_addr_value); // vtable ptr = 0x557dae962d48

  void *v_table_addr = (void*)v_table_addr_value;  // 把這8字節值轉為地址,即為虛函數表指針
  printf("vtable ptr = %p\n", v_table_addr); // vtable ptr = 0x557dae761cd4

  long f1_addr_value = *(long*)v_table_addr;  // 虛函數表前8字節為f1()函數指針值
  printf("f1() ptr = 0x%lx\n", f1_addr_value);  // f1() ptr = 0x557dae761cd4
  Fun f1 = (Fun)f1_addr_value;  // 虛函數表內存第1個8字節值轉為函數指針
  f1();  // 輸出:virtual void Base::f1()

  long f2_addr_value = *(long*)((char*)v_table_addr + 8);  // 虛函數表8-16字節為f2()函數指針值
  printf("f2() ptr = 0x%lx\n", f2_addr_value);  // f2() ptr = 0x557dae761d0c
  Fun f2 = (Fun)f2_addr_value;  // 虛函數表內存第2個8字節值轉為函數指針
  f2();  // 輸出:virtual void Base::f2()

  long f3_addr_value = *(long*)((char*)v_table_addr + 16);  // 虛函數表前16-24字節為f3()函數指針值
  printf("f3() ptr = 0x%lx\n", f3_addr_value);  // f3() ptr = 0x557dae761d44
  Fun f3 = (Fun)f3_addr_value;  // 虛函數表內存第3個8字節值轉為函數指針
  f3();  // virtual void Base::f3()

  return 0;
}

通過上述代碼的輸出結果可以驗證上圖的內存模型。

 

繼承基類重寫虛函數

現在定義一個繼承類Derived,重寫了f1()函數,也就是覆蓋掉了Base類中的函數f1()。同時又新增了虛擬函數f4()。

class Base {
public:
  virtual void f1(){
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }

  virtual void f2(){
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }

  virtual void f3(){
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }
};

class Derived : public Base
{
public:
  virtual void f1() override {
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }

  virtual void f4() {
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }
};

通過上一節類似的代碼可以驗證new Derived()其內存模型為
聊一聊C++虛函數表的問題
由此可以得出以下結論:

  1. 虛函數按照其聲明順序放于表中。
  2. 父類的虛函數在子類的虛函數前面。
  3. 覆蓋的函數放到了虛函數表中原來父類虛函數的位置。
  4. 沒有被覆蓋的虛函數函數位置不變。

繼承N個基類就有N個虛函數表,接下來使用代碼去驗證。

有3個基類Base1,Base2, Base3,都有兩個虛函數f1()、f2()。最后Derived 類繼承這3個基類。并重寫f1()函數,新增f4()函數。

class Base1 {
public:
  virtual void f1() {
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }

  virtual void f2() {
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }
};

class Base2 {
public:
  virtual void f1() {
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }

  virtual void f2() {
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }
};

class Base3 {
public:
  virtual void f1() {
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }

  virtual void f2() {
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }
};

class Derived : public Base1, public Base2, public Base3 {
public:
  void f1() override {
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }

  virtual void f4() {
      std::cout << __PRETTY_FUNCTION__ << std::endl;
  }
};

此時,sizeof(Derived) 等于24,可以基本確定類實例中有3個虛函數表指針。
下面通過代碼來檢查一下內存數據。

int main() {
  typedef void(*Fun)();

  std::cout << sizeof(Derived) << std::endl; // 24

  Derived *d = new Derived();
  printf("b ptr = %p\n", d); // b ptr = 0x5624201d9280

  long v_table1_addr_value = *(long *) d;  // 第1個虛函數表地址值
  printf("vtable1 ptr = 0x%lx\n", v_table1_addr_value); // vtable1 ptr = 0x56241e42ac48

  long b1f1_addr_value = *(long *) v_table1_addr_value;
  printf("b1f1() ptr = 0x%lx\n", b1f1_addr_value); // b1f1() ptr = 0x56241e22a170
  Fun b1f1 = (Fun) b1f1_addr_value;
  b1f1(); // virtual void Derived::f1()

  long b1f2_addr_value = *((long *) v_table1_addr_value + 1);
  printf("b1f2() ptr = 0x%lx\n", b1f2_addr_value); // b1f2() ptr = 0x56241e22a058
  Fun b1f2 = (Fun) b1f2_addr_value;
  b1f2(); // virtual void Base1::f2()

  long b1f3_addr_value = *((long *) v_table1_addr_value + 2);
  printf("b1f3() ptr = 0x%lx\n", b1f3_addr_value); // b1f3() ptr = 0x56241e22a1b4
  Fun b1f3 = (Fun) b1f3_addr_value;
  b1f3(); // virtual void Derived::f3()

  long v_table2_addr_value = *((long *) d + 1); // 類實例內存第2個8字節為 第2個虛函數表地址值
  printf("vtable2 ptr = 0x%lx\n", v_table2_addr_value); // vtable2 ptr = 0x56241e42ac70

  long b2f1_addr_value = *(long *) v_table2_addr_value;
  printf("b2f1() ptr = 0x%lx\n", b2f1_addr_value); // b2f1() ptr = 0x56241e22a1ad
  Fun b2f1 = (Fun) b2f1_addr_value;
  b2f1(); // virtual void Derived::f1()

  long b2f2_addr_value = *((long *) v_table2_addr_value + 1);
  printf("b2f2() ptr = 0x%lx\n", b2f2_addr_value); // b2f2() ptr = 0x56241e22a0c8
  Fun b2f2 = (Fun) b2f2_addr_value;
  b2f2(); // virtual void Base2::f2()

  long b2f3_addr_value = *((long *) v_table2_addr_value + 2);
  printf("b2f3() ptr = 0x%lx\n", b2f3_addr_value); // b2f3() ptr = 0xfffffffffffffff0


  long v_table3_addr_value = *((long *) d + 2); // 類實例內存第3個8字節為 第3個虛函數表地址值
  printf("vtable3 ptr = 0x%lx\n", v_table3_addr_value); // vtable3 ptr = 0x56241e42ac90

  long b3f1_addr_value = *(long *) v_table3_addr_value;
  printf("b3f1() ptr = 0x%lx\n", b3f1_addr_value); // b3f1() ptr = 0x56241e22a1a7
  Fun b3f1 = (Fun) b3f1_addr_value;
  b3f1(); // virtual void Derived::f1()

  long b3f2_addr_value = *((long *) v_table3_addr_value + 1);
  printf("b3f2() ptr = 0x%lx\n", b3f2_addr_value); // b3f2() ptr = 0x56241e22a138
  Fun b3f2 = (Fun) b3f2_addr_value;
  b3f2(); // virtual void Base3::f2()

  return 0;
}

根據上述代碼輸出結果,可以畫出下面內存模型。

聊一聊C++虛函數表的問題

由此可以得出以下結論:

  1. 有幾個基類就有幾個虛函數表,且實例中虛函數表地址值存儲順序就是基類繼承順序。
  2. 繼承類新增的虛函數f3()排在第一個虛函數表中,且在基類虛函數后面。
  3. 繼承類中重寫基類的虛函數f1(),在每個虛函數表中都覆蓋相應的虛函數。

 

尋找被覆蓋的虛函數

Derived 類重寫基類Base的f1()函數后,那如果想調用基類的被覆蓋的虛函數的話,就需要明確類名字調用。

Derived *d = new Derived();
  d->f1();  // virtual void Derived::f1()
  d->Base::f1();  // virtual void Base::f1()

內存空間中繼承類重寫的函數存在于虛函數表中原函數的位置,那么原虛函數的位置在哪呢?

到此這篇關于聊一聊C++虛函數表的問題的文章就介紹到這了,更多相關C++虛函數表內容請搜索服務器之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持服務器之家!

原文鏈接:https://blog.csdn.net/shaosunrise/article/details/120803945

延伸 · 閱讀

精彩推薦
  • C/C++C++之重載 重定義與重寫用法詳解

    C++之重載 重定義與重寫用法詳解

    這篇文章主要介紹了C++之重載 重定義與重寫用法詳解,本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內容,需要的朋友可以參考下...

    青山的青6062022-01-04
  • C/C++c++ 單線程實現同時監聽多個端口

    c++ 單線程實現同時監聽多個端口

    這篇文章主要介紹了c++ 單線程實現同時監聽多個端口的方法,幫助大家更好的理解和學習使用c++,感興趣的朋友可以了解下...

    源之緣11542021-10-27
  • C/C++詳解c語言中的 strcpy和strncpy字符串函數使用

    詳解c語言中的 strcpy和strncpy字符串函數使用

    strcpy 和strcnpy函數是字符串復制函數。接下來通過本文給大家介紹c語言中的strcpy和strncpy字符串函數使用,感興趣的朋友跟隨小編要求看看吧...

    spring-go5642021-07-02
  • C/C++深入理解goto語句的替代實現方式分析

    深入理解goto語句的替代實現方式分析

    本篇文章是對goto語句的替代實現方式進行了詳細的分析介紹,需要的朋友參考下...

    C語言教程網7342020-12-03
  • C/C++C/C++經典實例之模擬計算器示例代碼

    C/C++經典實例之模擬計算器示例代碼

    最近在看到的一個需求,本以為比較簡單,但花了不少時間,所以下面這篇文章主要給大家介紹了關于C/C++經典實例之模擬計算器的相關資料,文中通過示...

    jia150610152021-06-07
  • C/C++學習C++編程的必備軟件

    學習C++編程的必備軟件

    本文給大家分享的是作者在學習使用C++進行編程的時候所用到的一些常用的軟件,這里推薦給大家...

    謝恩銘10102021-05-08
  • C/C++C語言中炫酷的文件操作實例詳解

    C語言中炫酷的文件操作實例詳解

    內存中的數據都是暫時的,當程序結束時,它們都將丟失,為了永久性的保存大量的數據,C語言提供了對文件的操作,這篇文章主要給大家介紹了關于C語言中文件...

    針眼_6702022-01-24
  • C/C++C語言實現電腦關機程序

    C語言實現電腦關機程序

    這篇文章主要為大家詳細介紹了C語言實現電腦關機程序,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下...

    xiaocaidayong8482021-08-20
主站蜘蛛池模板: 精品香蕉视频 | 日韩中文视频 | 精品自拍视频 | 精品久久精品 | 中文在线观看www | 犬夜叉在线观看 | 久草视频免费看 | 综合五月网 | 久久网站热最新地址 | 精精国产xxxx视频在线野外 | 天天亚洲综合 | 一区二区三区免费 | 欧美一级特黄aaaaaaa在线观看 | 日本久久香蕉 | 亚洲精品成人天堂一二三 | 一级爱 | 精品黄色一级片 | 亚洲成人精品视频 | 精品久久久久一区二区国产 | 91亚洲精品 | av在线网址观看 | 亚洲高清免费视频 | 午夜伦4480yy私人影院 | 久久精彩| 中文字幕一区二区三区乱码在线 | 国产中文字幕在线免费观看 | 好看的一级毛片 | 免费av在线 | 精品国产乱码久久久久久影片 | 国产一区二区资源 | www一区二区 | 草草浮力影院 | 亚洲精品在线播放 | 久久99精品久久久久久园产越南 | 日韩成人免费 | 一区二区av在线 | 欧美视频免费看 | 精品伊人久久 | 精品小视频 | 一级毛片av | 美女午夜影院 |