亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

詳談C++中虛基類在派生類中的內存布局

2020-05-23 13:57:25
字體:
來源:轉載
供稿:網友

今天重溫C++的知識,當看到虛基類這點的時候,那時候也沒有太過追究,就是知道虛基類是消除了類繼承之間的二義性問題而已,可是很是好奇,它是怎么消除的,內存布局是怎么分配的呢?于是就深入研究了一下,具體的原理如下所示:

在C++中,obj是一個類的對象,p是指向obj的指針,該類里面有個數據成員mem,請問obj.mem和p->mem在實現和效率上有什么不同。

答案是:只有一種情況下才有重大差異,該情況必須滿足以下3個條件:

(1)、obj 是一個虛擬繼承的派生類的對象

(2)、mem是從虛擬基類派生下來的成員

(3)、p是基類類型的指針

當這種情況下,p->mem會比obj.mem多了兩個中間層。(也就是說在這種情況下,p->mem比obj.mem要明顯的慢)

WHY?

如果好奇心比較重的話,請往下看 :

1、虛基類的使用,和為多態而實現的虛函數不同,是為了解決多重繼承的二義性問題。

舉例如下:

class A{public:  int a;};class B : virtual public A{public:  int b;};class C :virtual public A{public:  int c; };class D : public B, public C{public:  int d;};

上面這種菱形的繼承體系中,如果沒有virtual繼承,那么D中就有兩個A的成員int a;繼承下來,使用的時候,就會有很多二義性。而加了virtual繼承,在D中就只有A的成員int a;的一份拷貝,該拷貝不是來自B,也不是來自C,而是一份單獨的拷貝,那么,編譯器是怎么實現的呢??

在回答這個問題之前,先想一下,sizeof(A),sizeof(B),sizeof(C),sizeof(D)是多少?(在32位x86的linux2.6下面,或者在vc2005下面)在linux2.6下面,結果如下:sizeof(A) = 4; sizeof(B) = 12; sizeof(C) = 12; sizeof(D) = 24;sizeof(B)為什么是12呢,那是因為多了一個指針(這一點和虛函數的實現一樣),那個指針是干嘛的呢?

那么sizeof(D)為什么是24呢?那是因為除了繼承B中的b,C中的c,A中的a,和D自己的成員d之外,還繼承了B,C多出來的2個指針(B和C分別有一個)。再強調一遍,D中的int a不是來自B也不是來自C,而是另外的一份從A直接靠過來的成員。

如果聲明了D的對象d: D d;

那么d的內存布局如下:

vb_ptr: 繼承自B的指針
int b: 繼承自B公有成員
vc_ptr:繼承自C的指針
int c: 繼承自C的共有成員
int d: D自己的公有成員
int a: 繼承自A的公有成員
 
那么以下的用法會發生什么事呢?

D dD;B *pb = &dD;pb->a;

上面說過,dD中的int a不是繼承自B的,也不是繼承自C的,那么這個B中的pb->a又會怎么知道指向的是dD內存中的第六項呢?

那就是指針vb_ptr的妙用了。原理如下:(其實g++3.4.3的實現更加復雜,我不知道是出于什么考慮,而我這里只說原理,所以把過程和內容簡單化了)

首先,vb_ptr指向一個整數的地址,里面放的整數是那個int a的距離dD開始處的位移(在這里vb_ptr指向的地址里面放的是20,以字節為單位)。編譯器是這樣做的:

首先,找到vb_ptr(這個不用找,因為在g++中,vb_ptr就是B*中的第一項,呵呵),然后取得vb_ptr指向的地址的內容(這個例子是20),最后把這個內容與指針pb相加,就得到pb->a的地址了。

所以說這種時候,用指針轉換多了兩個中間層才能找到基類的成員,而且是運行期間。

由此也可以推知dD中的vb_ptr和vc_ptr的內容都是一樣的,都是指向同一個地址,該地址就放20(在本例中)

如下的語句呢:

A *pa = &dD;pa->a = 4;

這個語句不用轉換了,因為編譯器在編譯期間就知道他把A中的成員插在dD中的那個地方了(在本例中是末尾),所以這個語句中的運行效率和dD.a是一樣的(至少也是差不多的)

這就是虛基類實現的基本原理。

注意的是:那些指針的位置和基類成員在派生類成員中的內存布局是不確定的,也就是說標準里面沒有規定inta必須要放在最后,只不過g++編譯器的實現而已。c++標準大概只規定了這套機制的原理,至于具體的實現,比如各成員的排放順序和優化,由各個編譯器廠商自己定~

非虛擬繼承:

在派生類對象里,按照繼承聲明順序依次分布基類對象,最后是派生類數據成員。

若基類聲明了虛函數,則基類對象頭部有一個虛函數表指針,然后是基類數據成員。

在基類虛函數表中,依次是基類的虛函數,若某個函數被派生類override,則替換為派生類的函數。

派生類獨有的虛函數被加在第一個基類的虛函數表后面。

虛擬繼承:

在派生類對象里,按照繼承聲明順序依次分布非虛基類對象,然后是派生類數據成員,最后是虛基類對象。

若基類聲明了虛函數,則基類對象頭部有一個虛函數表指針,然后是基類數據成員。

在基類虛函數表中,依次是基類的虛函數,若某個函數被派生類override,則替換為派生類的函數。

若直接從虛基類派生的類沒有非虛父類,且聲明了新的虛函數,則該派生類有自己的虛函數表,在該派生類頭部;否則派生類獨有的虛函數被加在第一個非虛基類的虛函數表后面。

直接從虛基類派生的類內部還有一個虛基類表指針(一個隱藏的“虛基類表指針”成員,指向一個虛基類表),在數據成員之前,非虛基類對象之后(若有的話)。

虛基類表中第一個值是該虛基類表到派生類起始地址的偏移;之后的值依次是該派生類的虛基類到該表位置的地址偏移(虛基類對象的地址與派生類的“虛基類表指針”之間的偏移量)。

對于虛函數表指針和虛基類表指針:

當單繼承且非虛繼承時:每個含有虛函數的表只有一個虛函數表,所以只需要一個虛表指針即可;

當多繼承且非虛繼承時:一個子類有幾個父類則會有幾個虛函數表,所以就有和父類個數相同的虛表指針來標識;

總之,當時非虛繼承時,不需要額外增加虛函數表指針。

當虛繼承時:無論是單虛繼承還是多虛繼承,需要有一個虛基類表來記錄虛繼承關系,所以此時子類需要多一個虛基類表指針;而且只需要一個即可。

當虛繼承時可能出現一個類中持有多個虛函數表的情況:無論是單虛繼承還是多虛繼承,

如果子類沒有構造函數和析構函數,且子類中的虛函數都是在父類中出現的虛函數,這個時候不需要增加任何虛表指針;只需要像多繼承那個持有父類個數的虛表指針來標識即可;

如果子類中含有構造函數或者析構函數或二者都有,則在子類中只要每出現一個父類中的虛函數則需要增加一個虛函數表指針來標識此類的虛函數表;

無論是否含有構造函數或者虛構函數,只要繼承都是虛繼承且出現了父類中沒有出現的虛函數,則在子類中需要再增加一個徐函數表指針;如果其中有一個是非虛繼承,則按照最省空間的原則,不需要增加虛函數表指針,因為這個時候可以和非虛基類共享一個虛函數表指針。

以上就是小編為大家帶來的詳談C++中虛基類在派生類中的內存布局全部內容了,希望大家多多支持VEVB武林網~


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
一区二区在线视频播放| 国产精品久久久久999| 成人激情电影一区二区| 亚洲天堂免费视频| 91久久精品一区| 国产一区二区美女视频| 国产一区二区日韩| 国产精品最新在线观看| 欧美日韩亚洲天堂| 青草成人免费视频| 亚洲国产中文字幕久久网| 欧美性极品xxxx娇小| 久久理论片午夜琪琪电影网| 精品爽片免费看久久| 日韩在线资源网| 欧美裸体男粗大视频在线观看| 热门国产精品亚洲第一区在线| 日韩中文字幕在线免费观看| 国产不卡在线观看| 久久久久99精品久久久久| 一个人看的www欧美| 日韩av免费在线| 精品久久久国产| 亚洲人成在线免费观看| 在线观看久久久久久| 97在线视频免费播放| 欧美极度另类性三渗透| 日韩精品中文字幕视频在线| 亚洲精品视频网上网址在线观看| 国产精品一区二区三区久久| 隔壁老王国产在线精品| 欧美亚洲激情视频| 成人免费自拍视频| 国产啪精品视频| 精品亚洲va在线va天堂资源站| 这里只有精品视频在线| 国产亚洲美女精品久久久| 91久久国产精品91久久性色| 91美女福利视频高清| 欧美一区二区三区艳史| 欧美激情在线视频二区| 91国内精品久久| 亚洲第一免费播放区| 久久久精品免费| 精品久久中文字幕久久av| 国产精品美女999| 96sao精品视频在线观看| 伊人久久久久久久久久久久久| 精品亚洲一区二区三区在线播放| 欧美丰满老妇厨房牲生活| 亚洲字幕一区二区| 欧美日韩国产精品专区| 91精品美女在线| 91在线免费视频| 日韩av在线播放资源| 久久成人人人人精品欧| 亚洲精品国产综合区久久久久久久| 欧美性xxxx| 91精品国产高清自在线| 国产精品va在线播放| 久久久99免费视频| 亚洲综合第一页| 日韩亚洲综合在线| 国产免费一区二区三区在线能观看| 国产小视频91| 欧美亚洲另类视频| 欧美丝袜一区二区三区| 免费成人高清视频| 欧美精品免费在线观看| 精品国产乱码久久久久久虫虫漫画| 久久精品亚洲热| 国产91精品在线播放| 九九九久久国产免费| 成人h猎奇视频网站| 亚洲级视频在线观看免费1级| 亚洲精品国产suv| 国产精品永久在线| 欧美激情影音先锋| 亚洲网站视频福利| 欧美激情一区二区三区在线视频观看| 亚洲精品白浆高清久久久久久| 欧美在线视频免费| 亚洲精品成a人在线观看| 亚洲成人精品久久久| 国产精品香蕉av| 亚洲欧洲在线看| 日韩成人在线视频| 亚洲成人av片| 在线观看欧美成人| 国产一区二区三区直播精品电影| 亚洲一区亚洲二区亚洲三区| 欧美日韩中文字幕在线视频| 亲子乱一区二区三区电影| 国产成人综合精品| 91精品久久久久久久久不口人| 欧美日韩精品在线| 91精品久久久久久久久久久| 国产成人短视频| 亚洲日本aⅴ片在线观看香蕉| 亚洲最大激情中文字幕| 亚洲黄页网在线观看| 亚洲r级在线观看| 免费成人高清视频| 国产亚洲精品久久久久久| 亚洲乱亚洲乱妇无码| 欧美日韩亚洲成人| 欧美极品少妇全裸体| 色阁综合伊人av| 欧美又大又粗又长| 欧美劲爆第一页| 欧美激情精品久久久久久大尺度| 欧美俄罗斯乱妇| 日韩欧美中文字幕在线观看| 欧美疯狂性受xxxxx另类| 538国产精品一区二区免费视频| 欧美国产日本高清在线| 中文字幕日韩有码| 亚洲人成网站999久久久综合| 欧美激情2020午夜免费观看| 91在线精品播放| 中文字幕成人精品久久不卡| 亚洲小视频在线观看| 国产成人久久久精品一区| 亚洲人成77777在线观看网| 欧美电影免费观看高清完整| 欧美日韩一区二区三区在线免费观看| 亚洲欧美精品在线| 日韩欧美中文在线| 日韩视频免费大全中文字幕| 98精品国产高清在线xxxx天堂| 国产成人综合精品| 亚洲xxxx视频| 精品亚洲国产视频| 久久久久五月天| 日本中文字幕成人| 成人精品网站在线观看| 亚洲午夜性刺激影院| 午夜免费在线观看精品视频| 97在线观看视频国产| 国产亚洲欧洲高清一区| 97人人模人人爽人人喊中文字| 欧美大码xxxx| 久久久噜噜噜久久| 91欧美日韩一区| 97视频在线观看免费高清完整版在线观看| 8x拔播拔播x8国产精品| 69视频在线免费观看| 亚洲福利视频久久| 欧美老妇交乱视频| 成人国产精品一区| 色中色综合影院手机版在线观看| 国产精品亚洲美女av网站| 精品久久久久久亚洲精品| 中文字幕欧美日韩| 91国语精品自产拍在线观看性色| 欧美国产日韩一区二区| 日韩欧美国产免费播放| 9.1国产丝袜在线观看| 国产欧美日韩精品专区| 成人免费自拍视频| 欧美一级片久久久久久久| 啊v视频在线一区二区三区| 狠狠色噜噜狠狠狠狠97| www.亚洲成人|