内存对齐

内存对齐指的是结构体中对结构成员内存的一系列调整。通过调整offset位置,减少读取结构成员数据需要的CPU-> 内存读取次数。任何对象数据在底层实际上是结构体。

对齐规则

1、数据成员对齐规则:结构(struct)(或联合(union))的数据成员,第一个数据成员放在offset为0的地方,以后每个数据成员的对齐按照#pragma pack指定的数值和这个数据成员自身长度中,比较小的那个进行

2、结构(或联合)的整体对齐规则:在数据成员完成各自对齐之后,结构(或联合)本身也要进行对齐,对齐将按照#pragma pack指定的数值和结构(或联合)最大数据成员长度中,比较小的那个进行

3、结合1、2可推断:当#pragma pack的n值等于或超过所有数据成员长度的时候,这个n值的大小将不产生任何效果。

是不是有一点眼晕?我自己总结的计算方法如下:

内存对齐原则 (简化版):

  1. 找到最大的基础类型成员 , 将其内存长度作为基准
  2. 每个成员的offset,需要是成员自己类型的整数倍
  3. 放完之后,struct自身的数据需要为基准的整数倍

让我们看一个简单的数据:

struct Struct1 {
    double a; // 8
    char b;   // 1
    int c;    // 4
    short d;  // 2
}struct1;

1. 最长的元素是double,基准为8
2.
struct Struct1 {
    double a; // 长度为8 offset:0,0是8的整数倍,所以 a存放位置为 (0 - 7)
    char b;   //  长度为1 offset: 8  8是 1的整数倍,所以 b存放位置为 (8)
    int c;    //  长度为4 offset: 9  9不是4的整数倍,offset后移到 12 ,所以 c存放位置为 (12 - 15)
    short d;  //  长度为2 offset: 16  16是2的整数倍,所以 d存放位置为 (16 - 17)
}struct1;

3. 最终数据成员的整体内存占用位置为 (0 - 17),18个字节。18不是基准 8 的整数倍,因此填充到24.

最终 sizeof(struct1) 的大小为 24

如果我们稍微调整下数据顺序:

struct Struct2 {
    double a; // 8
// 这里调整了 b,c的顺序
    int b;    // 4
    char c;   // 1
    short d;  // 2
}struct2;

1. 最长的元素是double,基准为8
2.
struct Struct2 {
    double a; // 长度为8 offset:0,0是8的整数倍,所以 a存放位置为 (0 - 7)
    int b;   //  长度为4 offset: 8  8是 4的整数倍,所以 b存放位置为 (8 - 11)
    char c;    //  长度为1 offset:12  12是1的整数倍,所以 c存放位置为 (12)
    short d;  //  长度为2 offset: 13  13不是2的整数倍,offset后移到 14 所以 d存放位置为 (14 - 15)
}struct2;

3. 最终数据成员的整体内存占用位置为 (0 - 15),16个字节。16是基准 8 的整数倍,因此不需要填充。

最终 sizeof(struct2) 的大小为 16

成员为数组的情况:

内存对齐原则 ( ex1):

  1. 找到最大的基础类型成员 , 将其内存长度作为基准 (如果成员为数组,基准不按数组总长,而是按照数组元素类型来计算)
struct Struct3 {
    char a; // 1 (0)
    int b[2]; // b0 b1
    double c; //8
}struct3;

1. 最长的元素是double,基准为8
2.
struct Struct3 {
    char a; // 长度为1 offset:0 0是1的整数倍,所以 a存放位置为 (0)
    int b[2]; // b0  长度为4  0ffset:1  1不是4的整数倍,b0的offest后移到4,所以 b0存放位置为 (4 - 7)b1存放位置为 (8 - 11)
   double c; //长度为8 offset: 12 12不是8的整数倍,offset后移到 16 所以 d存放位置为 (16 - 23)
}struct3;

3. 最终数据成员的整体内存占用位置为 (0 - 23),24个字节。24是基准 8 的整数倍,因此不需要填充。

最终 sizeof(struct3) 的大小为 24

其实数组写法很类似下面的写法:

struct Struct3 {
    char a; 
    int b0;
    int b1;
    double c;
}struct3;

只要调整对了数组第一个元素的offset,数组后续元素自然可以对齐。

成员为struct的情况:

  1. 找到最大的基础类型成员 , 将其内存长度作为基准 (如果成员为struct,基准不按子struct总长,而是max(子struct基准,剩余成员内存长度))
  2. 每个成员的offset,需要是成员自己类型的整数倍 (如果成员为struct, 只需是 成员struct 基准的整数倍而不是总长的整数倍)
struct Struct1 {
    double a; // 8
    char b;   // 1
    int c;    // 4
    short d;  // 2
}struct1;


struct Struct4 {
    int b;   // 4 // (0-3)
    char c;    // 1 //  (4)
    struct Struct1 s1; // a:8 (8 - 15) b:1 (16) c:4 (20 - 23) d:2 (24-25) (8 - 25) 18字节 ->
    short d;  // 2 // 38 (36 - 37)
}struct4;

1. Struct1的基准为8,除Struct1 s1外,Struct4最长的元素是int 长为4,基准为max(8,4) = 8

2. 

struct Struct4 {
    int b;   // 4 // offset:0 ,存放位置: (0-3)
    char c;    // 1 //  offset:4,存放位置: (4)
    struct Struct1 s1; 
/**  a: 长度: 8 offset:5 -> 8  存放位置: (8 - 15)
      b:长度: 1  offset:16  存放位置:(16)
      c: 长度: 4 offset:17 -> 20 存放位置: (20 - 23) 
      d: 长度:  2 offset:24  存放位置: (24-25) 
      s1 总体位置:(8 - 25) 18字节 -> 填充 (8 - 31)24字节
**/
    short d;  // 2 // offset:32,存放位置: (32 - 33)
}struct4;

3. 总体位置 (0 - 33) 长度为 34,不是8的倍数,填充到 40 

最终 sizeof(struct4) 的大小为 40

附表:

  1. 基础数据表


    image.png

2.内存分布查看方法:

po &struct4 -> 0x00000001014cc600

x/8gx 0x00000001014cc600
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335