日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 >

C++中虚函数的理解,以及简单继承情况下的虚函数的表!

發布時間:2023/12/15 40 豆豆
生活随笔 收集整理的這篇文章主要介紹了 C++中虚函数的理解,以及简单继承情况下的虚函数的表! 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

面向對象的三大特征=封裝性+繼承性+多態性
封裝=將客觀事物抽象成類,每個類對自身的數據和方法實行權限的控制
繼承=實現繼承+可視繼承+接口繼承
多態=將父類對象設置成為和一個或者更多它的子對象相等的技術,
用子類對象給父類對象賦值之后,
父類對象就可以根據當前賦值給它的子對象的特性一不同的方式運作

C++的空類有哪些成員函數
1.缺省構造函數
2.缺省拷貝構造函數
3.缺省析構函數
4.缺省賦值運算符
5.缺省取址運算符
6.缺省取址運算符const
PS:只有當實際使用的時候才會去使用這些類的成員函數的時候,編譯器才回去定義它們。

拷貝構造函數和賦值運算符重載有以下兩個不同之處:
1.拷貝構造函數生成的新的類對象,而賦值運算符不能;
2.由于拷貝構造函數是直接構造一個新的類對象,所以在初始化這個對象之前不用
檢驗源對象是否和新建對象相同。而賦值運算符則需要這個操作,另外賦值運算
中如果原來的對象中有內存分配要先把內存釋放掉。
PS:當類中有指針類型的成員變量時,一定要重寫拷貝構造函數和賦值運算符,不要使用默認的。

#include <iostream> #include <stdio.h>using namespace std; #ifndef __GNUC__ #define __attribute__(x) #endif // __GNUC____attribute__((constructor)) void before_main() {cout << __FUNCTION__ << endl;cout << "這是在Main函數運行前執行的!" << endl; } __attribute__((destructor)) void after_main() {cout << __FUNCTION__ << endl;cout << "這是在Main函數結束后執行的!" << endl; }

對C++了解的人都知道虛函數(Virtual Function)是通過一張虛函數表(Virtual Table)來實現的。
在這個表中,主是要一個類的虛函數的地址表,這張表解決了繼承、覆蓋的問題,保證真實反應實際的函數。
這樣,在有虛函數的類的實例中這個表被分配在了這個實例的內存中,所以當用父類的指針來操作一個子類的時候,
這張虛函數表就顯得由為重要了,它就像一個地圖一樣,指明了實際所應該調用的函數。
在C++標準規格說明書中說到,編譯器必需要保證虛函數表的指針存在于對象實例中最前面的位置,這是為了保證正確取到虛函數的偏移量。
這意味著我們通過對象實例的地址得到這張虛函數表,然后就可以遍歷其中函數指針,并調用相應的函數。

//訪問基類的私有虛函數 class A { public:virtual void g(){cout << "A:g()" << endl;}virtual void f(){cout << "A::f()" << endl;}virtual void h(){cout << "A::h()" << endl;} }; /* 這類的虛函數表的排布方式是 vptr|| &(A::g())|_ _ _ _ \&(A::f())/&(A::h()) */class B:public A { public:virtual void g(){cout << "B::g()" << endl;}virtual void h(){cout << "B::h()" << endl;} }; /* 這個類的虛函數表是 vptr|| &(A::g())-->&(B::g())|_ _ _ _ \&(A::f())/&(A::h())-->&(B::h()) */ /* Mark 一下:每一個具有虛函數的類都有一個虛函數表VTABLE,里面按在類中聲明的虛函數的順序存放著虛函數的地址,這個虛函數表VTABLE是這個類的所有對象所共有的,也就是說無論用戶聲明了多少個類對象,但是這個VTABLE虛函數表只有一個。在每個具有虛函數的類的對象里面都有一個VPTR虛函數指針,這個指針指向VTABLE的首地址,每個類的對象都有這么一種指針。 */ class C { public:C() { cout << "C Constructor !" << endl; }~C() { cout << "C Destroy !" << endl; } };//會在main函數調用之前執行的函數 static C s_c; typedef void (*PFUNC)(void);class AA { private:int a;double b; public:virtual void func(){cout << "AA::func()" << endl;} }; /* 在內存中的存儲格式是 VPTR-->a-->b||__\ &func()/ &func() */int main() {cout << __FUNCTION__ << endl;cout << "Hello world!" << endl;/*很多人找不到vptr這個指針,其實這個B類繼承自A類,且沒有自己的成員變量,所以可以得出,B類的事例對象中的只有vptr這一個指針了。*/B b;PFUNC pf;//先打印出虛函數表的地址cout << "虛函數表地址:" << (int*)(&b) << endl;cout << "虛函數表第一個函數地址:" << (int*)(*(int*)(&b)) << endl;for(int i=0;i<3;i++){pf=(PFUNC)*((int*)*(int*)(&b)+i);pf();}return 0; }

總結

以上是生活随笔為你收集整理的C++中虚函数的理解,以及简单继承情况下的虚函数的表!的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。