C++對(duì)象內(nèi)存分布詳解(包括字節(jié)對(duì)齊和虛函數(shù)表)
1、C++對(duì)象的內(nèi)存分布和虛函數(shù)表:
C++對(duì)象的內(nèi)存分布和虛函數(shù)表注意,對(duì)象中保存的是虛函數(shù)表指針,而不是虛函數(shù)表,虛函數(shù)表在編譯階段就已經(jīng)生成,同類的不同對(duì)象中的虛函數(shù)指針指向同一個(gè)虛函數(shù)表,不同類對(duì)象的虛函數(shù)指針指向不同虛函數(shù)表。
2、何時(shí)進(jìn)行動(dòng)態(tài)綁定:
(1)每個(gè)類對(duì)象在被構(gòu)造時(shí)不用去關(guān)心是否有其他類從自己派生,也不需要關(guān)心自己是否從其他類派生,只要按照一個(gè)統(tǒng)一的流程:在自身的構(gòu)造函數(shù)執(zhí)行之前把自己所屬類(即當(dāng)前構(gòu)造函數(shù)所屬的類)的虛函數(shù)表的地址綁定到當(dāng)前對(duì)象上(一般是保存在對(duì)象內(nèi)存空間中的前4個(gè)字節(jié))。因?yàn)閷?duì)象的構(gòu)造是從最基類部分(比如A<-B<-C,A是最基類,C是最派生類)開始構(gòu)造,一層一層往外構(gòu)造中間類(B),最后構(gòu)造的是最派生類(C),所以最終對(duì)象上綁定的就自然而然就是最派生類的虛函數(shù)表。
(2)析構(gòu)函數(shù)的調(diào)用跟構(gòu)造函數(shù)的調(diào)用順序是相反的,它從最派生類的析構(gòu)函數(shù)開始的。也就是說(shuō)當(dāng)基類的析構(gòu)函數(shù)執(zhí)行時(shí),派生類的析構(gòu)函數(shù)已經(jīng)執(zhí)行過,派生類中的成員數(shù)據(jù)被認(rèn)為已經(jīng)無(wú)效(包括派生類對(duì)象中的虛表指針)。假設(shè)基類中虛函數(shù)調(diào)用能調(diào)用得到派生類的虛函數(shù),那么派生類的虛函數(shù)將訪問一些已經(jīng)“無(wú)效”的數(shù)據(jù),所帶來(lái)的問題和訪問一些未初始化的數(shù)據(jù)一樣。而同樣,我們可以認(rèn)為在析構(gòu)的過程中,虛函數(shù)表也是在不斷變化的,不斷解綁定。
因此,在基類構(gòu)造函數(shù)或者析構(gòu)函數(shù)中調(diào)用虛函數(shù),并不會(huì)綁定到派生類的實(shí)現(xiàn)上,因?yàn)樵谶@兩個(gè)函數(shù)執(zhí)行時(shí)虛函數(shù)表指針指向的是基類的虛函數(shù)表。
3、C++中類的大?。?/strong>
由 1 可知,C++對(duì)象中只保存非靜態(tài)數(shù)據(jù)成員,成員函數(shù)和靜態(tài)數(shù)據(jù)成員是存儲(chǔ)在靜態(tài)數(shù)據(jù)區(qū)的。
字節(jié)對(duì)齊(默認(rèn)):
1、VC規(guī)定各成員變量存放的起始地址相對(duì)于結(jié)構(gòu)的起始地址的偏移量必須為該變量的類型所占用的字節(jié)數(shù)的倍數(shù)。
2、VC為了確保結(jié)構(gòu)的大小為結(jié)構(gòu)的字節(jié)邊界數(shù)(即該結(jié)構(gòu)中占用最大空間的類型所占用的字節(jié)數(shù))的倍數(shù),所以在為最后一個(gè)成員變量申請(qǐng)空間后,還會(huì)根據(jù)需要自動(dòng)填充空缺的字節(jié)。
3、如果對(duì)齊字節(jié)數(shù)(#pragma pack(n)),那么
(1)各成員變量存放的起始地址相對(duì)于結(jié)構(gòu)的起始地址的偏移量必須為該變量的類型所占用的字節(jié)數(shù)和n的較小值的倍數(shù)。
(2)結(jié)構(gòu)的大小為結(jié)構(gòu)中占用最大空間的類型所占用的字節(jié)數(shù)和n的較小值的倍數(shù)。
class A {
double d;
static int i;
void f() { std::cout << "A::f" << std::endl; }
}; // 8 byte,只有double數(shù)據(jù)成員占8字節(jié),成員函數(shù)和靜態(tài)數(shù)據(jù)成員不在對(duì)象中,而是在靜態(tài)數(shù)據(jù)區(qū)
class B {
int i; //4
double j;//8
char k; //
}; // 24 byte,考慮字節(jié)對(duì)齊, 4 + 4 + 8 + 1 + 7, 藍(lán)色的4是為了滿足條件1,黑色的7是為了滿足條件2。如果指定4字節(jié)對(duì)齊,4 + 8 + 1 + 3
class C {
virtual void f() { std::cout << "C::f" << std::endl; }
}; // 4 byte,虛函數(shù)表指針占4個(gè)字節(jié)
class D {
};// 1 byte,沒有成員變量的結(jié)構(gòu)或類的大小為1,因?yàn)楸仨毐WC結(jié)構(gòu)或類的每一 個(gè)實(shí)例在內(nèi)存中都有唯一的地址
注:
1、如果有成員對(duì)象,直接把成員對(duì)象展開到外部對(duì)象中,然后按照字節(jié)對(duì)齊的規(guī)律求大小。
2、虛繼承的內(nèi)存分布為:虛類指針-》派生類成員數(shù)據(jù)-》基類成員數(shù)據(jù)。其對(duì)齊方案是:首先把派生類所有成員當(dāng)成一個(gè)嵌套結(jié)構(gòu)體形式,位于最下面的基類的數(shù)據(jù)成員要保證自己對(duì)齊(首地址整除自己的字節(jié)數(shù)),但是不用在最下面添加字節(jié)保證整體是邊界長(zhǎng)度的整數(shù)倍(因?yàn)榛惓蓡T共享,不能把派生類當(dāng)成一個(gè)整體)。
3、如果對(duì)象中有數(shù)組,可以把數(shù)組展開到對(duì)象中,然后按照字節(jié)對(duì)齊的規(guī)律求大小。
4、為什么要進(jìn)行字節(jié)對(duì)齊
計(jì)算機(jī)組成原理教導(dǎo)我們這樣有助于加快計(jì)算機(jī)的取數(shù)速度,否則就得多花指令周期了。為此,編譯器默認(rèn)會(huì)對(duì)結(jié)構(gòu)體進(jìn)行處理(實(shí)際上其它地方的數(shù)據(jù)變量也是如此),讓寬度為2的基本數(shù)據(jù)類型(short等)都位于能被2整除的地址上,讓寬度為4的基本數(shù)據(jù)類型(int等)都位于能被4整除的地址上,以此類推。比如有些平臺(tái)每次讀都是從偶地址開始,如果一個(gè)int型(假設(shè)為32位系統(tǒng))如果存放在偶地址開始的地方,那么一個(gè)讀周期就可以讀出這32bit,而如果存放在奇地址開始的地方,就需要2個(gè)讀周期,并對(duì)兩次讀出的結(jié)果的高低字節(jié)進(jìn)行拼湊才能得到該32bit數(shù)據(jù)。
備注:visual studio 2010是按照默認(rèn)方式進(jìn)行字節(jié)對(duì)齊的 32位gcc按照4字節(jié)最齊
以上就是小編為大家?guī)?lái)的C++對(duì)象內(nèi)存分布詳解(包括字節(jié)對(duì)齊和虛函數(shù)表)全部?jī)?nèi)容了,希望大家多多支持我們~
上一篇:C++ 類中有虛函數(shù)(虛函數(shù)表)時(shí) 內(nèi)存分布詳解
欄 目:C語(yǔ)言
下一篇:淺談#ifndef,#define,#endif的作用和用法
本文標(biāo)題:C++對(duì)象內(nèi)存分布詳解(包括字節(jié)對(duì)齊和虛函數(shù)表)
本文地址:http://www.jygsgssxh.com/a1/Cyuyan/1883.html
您可能感興趣的文章
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用函數(shù)刪除字符
- 04-02c語(yǔ)言沒有round函數(shù) round c語(yǔ)言
- 01-10深入理解C++中常見的關(guān)鍵字含義
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10如何查看進(jìn)程實(shí)際的內(nèi)存占用情況詳解
- 01-10c++中inline的用法分析
- 01-10用C++實(shí)現(xiàn)DBSCAN聚類算法
- 01-10全排列算法的非遞歸實(shí)現(xiàn)與遞歸實(shí)現(xiàn)的方法(C++)
- 01-10C++大數(shù)模板(推薦)
- 01-10淺談C/C++中的static與extern關(guān)鍵字的使用詳解


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫分段 用c語(yǔ)言表示分段
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)
- 04-02c語(yǔ)言編寫函數(shù)冒泡排序 c語(yǔ)言冒泡排
- 04-02c語(yǔ)言沒有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求
隨機(jī)閱讀
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-10delphi制作wav文件的方法
- 04-02jquery與jsp,用jquery
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10C#中split用法實(shí)例總結(jié)


