中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

如何解決C++結構體內存對齊計算問題

發布時間:2021-09-16 09:47:32 來源:億速云 閱讀:104 作者:柒染 欄目:編程語言

這篇文章將為大家詳細講解有關如何解決C++結構體內存對齊計算問題,文章內容質量較高,因此小編分享給大家做個參考,希望大家閱讀完這篇文章后對相關知識有一定的了解。

編譯環境:vs2015

對齊原則:

      原則1:數據成員對齊規則:結構(struct)(或聯合(union))的數據成員,第一個數據成員放在offset為0的地方,以后每個數據成員的對齊按照#pragma pack指定的數值和這個數據成員自身長度中,比較小的那個進行。

      原則2:結構(或聯合)的整體對齊規則:在數據成員完成各自對齊之后,結構(或聯合)本身也要進行對齊,對齊將按照#pragma pack指定的數值和結構(或聯合)最大數據成員長度中,比較小的那個進行。

      原則3:結構體作為成員:如果一個結構里有某些結構體成員,則結構體成員要從其內部最大元素大小的整數倍地址開始存儲。

默認對齊值:

Linux 默認#pragma pack(4)

window 默認#pragma pack(8)

注:可以通過預編譯命令#pragma pack(n) ,n=1,2,4,8,16來改變這一系數,其中的n就是指定的“對齊系數”。

例一:一字節對齊

第一步: 成員數據對齊

#pragma pack(1)
struct AA {
 int a; //長度4 < 1 按1對齊;偏移量為0;存放位置區間[0,3]
 char b; //長度1 = 1 按1對齊;偏移量為4;存放位置區間[4]
 short c; //長度2 > 1 按1對齊;偏移量為5;存放位置區間[5,6]
 char d; //長度1 = 1 按1對齊;偏移量為6;存放位置區間[7]
 //整體存放在[0~7]位置區間中,共八個字節。
};
#pragma pack()

第二步: 整體對齊

整體對齊系數 = min((max(int,short,char), 1) = 1,所以不需要再進行整體對齊。整體大小就為8。

圖示如下:

如何解決C++結構體內存對齊計算問題

例二:二字節對齊

第一步: 成員數據對齊

#pragma pack(2)
struct AA {
 int a; //長度4 > 2 按2對齊;偏移量為0;存放位置區間[0,3]
 char b; //長度1 < 2 按1對齊;偏移量為4;存放位置區間[4]
 short c; //長度2 = 2 按2對齊;偏移量要提升到2的倍數6;存放位置區間[6,7]
 char d; //長度1 < 2 按1對齊;偏移量為7;存放位置區間[8];共九個字節
};
#pragma pack()

第二步: 整體對齊

整體對齊系數 = min((max(int,short,char), 2) = 2,將9提升到2的倍數,則為10.所以最終結果為10個字節。

圖示如下:(X為補齊部分)

如何解決C++結構體內存對齊計算問題

例三:四字節對齊

第一步: 成員數據對齊

#pragma pack(4)
struct AA {
 int a; //長度4 = 4 按4對齊;偏移量為0;存放位置區間[0,3]
 char b; //長度1 < 4 按1對齊;偏移量為4;存放位置區間[4]
 short c; //長度2 < 4 按2對齊;偏移量要提升到2的倍數6;存放位置區間[6,7]
 char d; //長度1 < 4 按1對齊;偏移量為7;存放位置區間[8];總大小為9
};
#pragma pack()

第二步: 整體對齊

整體對齊系數 = min((max(int,short,char), 4) = 4,將9提升到4的倍數,則為12.所以最終結果為12個字節。

圖示如下:(X為補齊部分)

如何解決C++結構體內存對齊計算問題

例三:八字節對齊

第一步: 成員數據對齊

#pragma pack(8)
struct AA {
 int a; //長度4 < 8 按4對齊;偏移量為0;存放位置區間[0,3]
 char b; //長度1 < 8 按1對齊;偏移量為4;存放位置區間[4]
 short c; //長度2 < 8 按2對齊;偏移量要提升到2的倍數6;存放位置區間[6,7]
 char d; //長度1 < 8 按1對齊;偏移量為7;存放位置區間[8],總大小為9
};
#pragma pack()

第二步: 整體對齊

整體對齊系數 = min((max(int,short,char), 8) = 4,將9提升到4的倍數,則為12.所以最終結果為12個字節。圖示如上。

注:可以通過stddef.h庫中的offsetof宏來查看對應結構體元素的偏移量。

例四:結構體中包含結構體的運算

整體計算過程如下

struct EE
{
 int a; //長度4 < 8 按4對齊;偏移量為0;存放位置區間[0,3]
 char b; //長度1 < 8 按1對齊;偏移量為4;存放位置區間[4]
 short c; //長度2 < 8 按2對齊;偏移量由5提升到6;存放位置區間[6,7]
 //結構體內部最大元素為int,由于偏移量為8剛好是4的整數倍,所以從8開始存放接下來的struct FF
 struct FF
 {
 int a1; //長度4 < 8 按4對齊;偏移量為8;存放位置區間[8,11]
 char b1; //長度1 < 8 按1對齊;偏移量為12;存放位置區間[12]
 short c1; //長度2 < 8 按2對齊;偏移量為13,提升到2的倍數14;存放位置區間[14,15]
 char d1; //長度1 < 8 按1對齊;偏移量為16;存放位置區間[16]
 };
 //整體對齊系數 = min((max(int,short,char), 8) = 4,將內存大小由17補齊到4的整數倍20
 char d;  //長度1 < 8 按1對齊;偏移量為21;存放位置區間[21]
 //整體對齊系數 = min((max(int,short,char), 8) = 4,將內存大小由21補齊到4的整數倍24
};

圖示如下:

如何解決C++結構體內存對齊計算問題

例五:再來一個嵌套結構體的計算

整體計算過程如下

struct B {
 char e[2]; //長度1 < 8 按2對齊;偏移量為0;存放位置區間[0,1]
 short h; //長度2 < 8 按2對齊;偏移量為2;存放位置區間[2,3]
 //結構體內部最大元素為double,偏移量為4,提升到8,所以從8開始存放接下來的struct A
 struct A {
 int a; //長度4 < 8 按4對齊;偏移量為8;存放位置區間[8,11]
 double b; //長度8 = 8 按8對齊;偏移量為12,提升到16;存放位置區間16,23]
 float c; //長度4 < 8,按4對齊;偏移量為24,存放位置區間[24,27]
 };
 //整體對齊系數 = min((max(int,double,float), 8) = 8,將內存大小由28補齊到8的整數倍32
};

圖示如下:

如何解決C++結構體內存對齊計算問題

小結:當#pragma pack的n值等于或超過所有數據成員長度的時候,這個n值的大小將不產生任何效果。

關于如何解決C++結構體內存對齊計算問題就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

c++
AI

四平市| 龙口市| 商城县| 定远县| 屏东市| 芜湖县| 瑞昌市| 翁牛特旗| 莆田市| 临朐县| 灵寿县| 乐至县| 通江县| 雷州市| 博客| 南丰县| 安平县| 孟津县| 霍州市| 三穗县| 历史| 灵石县| 揭东县| 香格里拉县| 建瓯市| 中方县| 武功县| 息烽县| 庆阳市| 若尔盖县| 惠水县| 都匀市| 华安县| 乾安县| 涞水县| 天气| 盐津县| 长垣县| 克什克腾旗| 乌拉特后旗| 三原县|