教你5分鐘輕松搞定內(nèi)存字節(jié)對(duì)齊
寫出一個(gè)struct,然后sizeof,你會(huì)不會(huì)經(jīng)常對(duì)結(jié)果感到奇怪?sizeof的結(jié)果往往都比你聲明的變量總長(zhǎng)度要大,這是怎么回事呢?講講字節(jié)對(duì)齊吧.
/******************************分割線
如果體系結(jié)構(gòu)是不對(duì)齊的,A中的成員將會(huì)一個(gè)挨一個(gè)存儲(chǔ),從而sizeof(a)為11。顯然對(duì)齊更浪費(fèi)了空間。那么為什么要使用對(duì)齊呢?
體系結(jié)構(gòu)的對(duì)齊和不對(duì)齊,是在時(shí)間和空間上的一個(gè)權(quán)衡。對(duì)齊節(jié)省了時(shí)間。假設(shè)一個(gè)體系結(jié)構(gòu)的字長(zhǎng)為w,那么它同時(shí)就假設(shè)了在這種體系結(jié)構(gòu)上對(duì)寬度為w的數(shù)據(jù)的處理最頻繁也是最重要的。它的設(shè)計(jì)也是從優(yōu)先提高對(duì)w位數(shù)據(jù)操作的效率來(lái)考慮的。比如說(shuō)讀寫時(shí).............此處省略50萬(wàn)字
*******************************************************/
上面是你隨便google一下,人家就可以跟你解釋的,一大堆的道理,我們沒(méi)怎么多時(shí)間,討論為何要對(duì)齊.直入主題,怎么判斷內(nèi)存對(duì)齊規(guī)則,sizeof的結(jié)果怎么來(lái)的,請(qǐng)牢記以下3條原則:(在沒(méi)有#pragma pack宏的情況下,務(wù)必看完最后一行)
1:數(shù)據(jù)成員對(duì)齊規(guī)則:結(jié)構(gòu)(struct)(或聯(lián)合(union))的數(shù)據(jù)成員,第一個(gè)數(shù)據(jù)成員放在offset為0的地方,以后每個(gè)數(shù)據(jù)成員存儲(chǔ)的起始位置要從該成員大小或者成員的子成員大?。ㄖ灰摮蓡T有子成員,比如說(shuō)是數(shù)組,結(jié)構(gòu)體等)的整數(shù)倍開(kāi)始(比如int在32位機(jī)為4字節(jié),則要從4的整數(shù)倍地址開(kāi)始存儲(chǔ)。
2:結(jié)構(gòu)體作為成員:如果一個(gè)結(jié)構(gòu)里有某些結(jié)構(gòu)體成員,則結(jié)構(gòu)體成員要從其內(nèi)部最大元素大小的整數(shù)倍地址開(kāi)始存儲(chǔ).(struct a里存有struct b,b里有char,int ,double等元素,那b應(yīng)該從8的整數(shù)倍開(kāi)始存儲(chǔ).)
3:收尾工作:結(jié)構(gòu)體的總大小,也就是sizeof的結(jié)果,.必須是其內(nèi)部最大成員的整數(shù)倍.不足的要補(bǔ)齊.
等你看完此3條原則,2分鐘已經(jīng)過(guò)去,抓緊時(shí)間,實(shí)戰(zhàn)3分鐘:
typedef struct bb
{
int id; //[0]....[3]
double weight; //[8].....[15] 原則1
float height; //[16]..[19],總長(zhǎng)要為8的整數(shù)倍,補(bǔ)齊[20]...[23] 原則3
}BB;
typedef struct aa
{
char name[2]; //[0],[1]
int id; //[4]...[7] 原則1
double score; //[8]....[15]
short grade; //[16],[17]
BB b; //[24]......[47] 原則2
}AA;
int main()
{
AA a;
cout<<sizeof(a)<<" "<<sizeof(BB)<<endl;
return 0;
}
結(jié)果是
48 24
ok,上面的全看明白了,內(nèi)存對(duì)齊基本過(guò)關(guān).
再講講#pragma pack().
在代碼前加一句#pragma pack(1),你會(huì)很高興的發(fā)現(xiàn),上面的代碼輸出為
32 16
bb是4+8+4=16,aa是2+4+8+2+16=32;
這不是理想中的沒(méi)有內(nèi)存對(duì)齊的世界嗎.沒(méi)錯(cuò),#pragma pack(1),告訴編譯器,所有的對(duì)齊都按照1的整數(shù)倍對(duì)齊,換句話說(shuō)就是沒(méi)有對(duì)齊規(guī)則.
明白了不?
那#pragma pack(2)的結(jié)果又是多少呢?對(duì)不起,5分鐘到了,自己去測(cè)試吧.
ps:Vc,Vs等編譯器默認(rèn)是#pragma pack(8),所以測(cè)試我們的規(guī)則會(huì)正常;注意gcc默認(rèn)是#pragma pack(4),并且gcc只支持1,2,4對(duì)齊。套用三原則里計(jì)算的對(duì)齊值是不能大于#pragma pack指定的n值。
- 淺析C++字節(jié)對(duì)齊容易被忽略的兩個(gè)問(wèn)題
- 關(guān)于C++內(nèi)存中字節(jié)對(duì)齊問(wèn)題的詳細(xì)介紹
- C++中聲明類的class與聲明結(jié)構(gòu)體的struct關(guān)鍵字詳解
- c++中struct使用注意事項(xiàng)
- C++中關(guān)鍵字Struct和Class的區(qū)別
- C++中typedef 及其與struct的結(jié)合使用
- 淺析c與c++中struct的區(qū)別
- 深入C++中struct與class的區(qū)別分析
- C++ 關(guān)于STL中sort()對(duì)struct排序的方法
- 深入剖析C++中的struct結(jié)構(gòu)體字節(jié)對(duì)齊
相關(guān)文章
Qt實(shí)現(xiàn)字幕滾動(dòng)效果的示例代碼
這篇文章主要介紹了Qt如何利用QTimer實(shí)現(xiàn)字幕滾動(dòng)功能,并且可以實(shí)現(xiàn)自行更改文本內(nèi)容、自適應(yīng)文本大小、自由調(diào)整速度等功能,感興趣的可以學(xué)習(xí)一下2022-06-06C語(yǔ)言中的結(jié)構(gòu)體的入門學(xué)習(xí)教程
這篇文章主要介紹了C語(yǔ)言中的結(jié)構(gòu)體的入門學(xué)習(xí)教程,以struct語(yǔ)句定義的結(jié)構(gòu)體是C語(yǔ)言編程中的重要基礎(chǔ),需要的朋友可以參考下2015-12-12C++設(shè)計(jì)模式編程中Template Method模板方法模式的運(yùn)用
這篇文章主要介紹了C++設(shè)計(jì)模式編程中Template Method模板方法模式的運(yùn)用,講到了包括模板方法模式中的細(xì)分方法以及適用場(chǎng)景,需要的朋友可以參考下2016-03-03c語(yǔ)言實(shí)現(xiàn)基數(shù)排序解析及代碼示例
這篇文章主要介紹了c語(yǔ)言實(shí)現(xiàn)基數(shù)排序解析及代碼示例,具有一定借鑒價(jià)值,需要的朋友可以參考下。2017-12-12C++使用宏實(shí)現(xiàn)動(dòng)態(tài)庫(kù)加載
開(kāi)發(fā)的時(shí)候,有些項(xiàng)目不能靜態(tài)鏈接動(dòng)態(tài)庫(kù),需要程序運(yùn)行時(shí)加載動(dòng)態(tài)庫(kù)。本文將使用宏來(lái)實(shí)現(xiàn)動(dòng)態(tài)庫(kù)的加載,感興趣的小伙伴可以跟隨小編一起了解一下2022-12-12C++核心編程之占位參數(shù)和默認(rèn)參數(shù)
這篇文章主要介紹了C++核心編程之占位參數(shù)和默認(rèn)參數(shù),c++中函數(shù)的形參列表中的形參是可以有默認(rèn)值的,函數(shù)的形參列表里可以有占位參數(shù),用來(lái)占位,調(diào)用函數(shù)時(shí)必須填補(bǔ)位置。下面更多相關(guān)內(nèi)容的詳細(xì)介紹,需要的小伙伴可以參考一下2022-03-03