内存对齐

内存对齐规则:

  • 数据成员对齐规则
    结构(struct)(或联合(union))的数据成员,第一个数据成员放在offset(偏移)为0的地方,以后每个数据成员的对齐按照 #pragma pack 指定的数值和这个数据成员自身长度中,比较小的那个进行
  • 结构(或联合)的整体对齐规则
    在数据成员完成各自对齐之后,结构(或联合)本身也要进行对齐,对齐将按照 #pragma pack 指定的数值和结构(或联合)最大数据成员长度中,比较小的那个进行。
  • 结构体作为成员:
    如果一个结构里有某些结构体成员,则结构体成员要从其内部最大元素大小的整数倍地址开始存储。

#pragma pack(n) 对齐系数:

每个特定平台上的编译器都有自己的默认“对齐系数”(也叫对齐模数)。程序员可以通过预编译命令#pragma pack(n),n=1,2,4,8,16来改变这一系数,其中的n就是你要指定的“对齐系数”。其中,Xcode 中默认为#pragma pack(8)

系统对齐:

  • 在操作系统中对结构(struct)(或联合(union))的数据成员分配的内存是操作系统的对齐系数的倍数。例如IOS中的系统对齐系数是16。

演练:

1.

struct StructOne { 
                   长度        对齐      偏移     区间
    char   a;       1  < 8     1         0      [0]     
    double b;       8  = 8     8         8      [8,  15]
    int    c;       4  < 8     4         16     [16, 19]
    short  d;       2  < 8     2         20     [20, 21]
} MyStruct1;

解读:
1、每个数据成员的对齐按照 #pragma pack 指定的数值和这个数据成员自身长度中,比较小的那个进行
-- char   a 的自身长度为 1, 1 < 8, 按 1 对齐
-- double b 的自身长度为 8, 8 = 8, 按 8 对齐
-- int    c 的自身长度为 4, 4 < 8, 按 4 对齐
-- short  d 的自身长度为 2, 2 < 8, 按 2 对齐
2、第一个数据成员放在offset为0的地方
-- char   a 的偏移为 1
-- double b 的偏移为 8
-- int    c 的偏移为 4
-- short  d 的偏移为 2
3、整体将按照 #pragma pack 指定的数值和结构(或联合)最大数据成员长度中,比较小的那个进行。
min((max(int,short,char,double), 8) = 8,将 21 提升到 8 的倍数,则为 24,所以最终结果为 24 个字节

2.

struct StructOne { 
                  长度        对齐      偏移     区间
    double b;       8  = 8     8         0      [0,   7]     
    char   a;       1  < 8     1         8      [8]
    short  d;       2  < 8     2         10     [10, 11]
    int    c;       4  < 8     4         12     [12, 15]
} MyStruct1;

解读:
这个和演练1不同的是short d,由于它是 2 对齐,上一个偏移量为 9,
9不是2的整数倍,所以向上取整,到10,"short b" 偏移为10
整体对齐系数 = min((max(int,short,char,double), 8) = 8,将15提升到8的倍数,则为16,所以最终结果为16个字节

3. 结构体包含结构体:

struct EE
{                        长度        对齐      偏移     区间
    int     a;            4 < 8       4        0      [0, 3]
    char    b;            1 < 8       1        4      [4] 
    short   c;            2 < 8       2        6      [6, 7]
    struct FF //结构体内部最大元素为int ,由于偏移量为8刚好是4的整数倍,所以从8开始存放接下来的struct FF   
    {
        int   a1;         4 < 8       4        8      [8, 11]
        char  b1;         1 < 8       1        12     [12]      
        short c1;         2 < 8       2        14     [14, 15]    
        char  d1;         1 < 8       1        16     [16] 
    };// 内部结构体整体对齐:min(max(int, char, short), 8) = 4, 将内存大小由17补齐到4的整数倍20
    char d;               1 < 8       1        21     [21] 
    // 整体对齐系数 = min((max(int,short,char), 8) = 4,将内存大小由21补齐到4的整数倍24
};
struct B {             长度        对齐      偏移     区间
    char   e[2];        1 < 8      2        0      [0, 1]   
    short  h;           2 < 8      2        2      [2, 3]  
    struct A { //结构体内部最大元素为double ,由于偏移量为8刚好是4的整数倍,所以从8开始存放接下来的struct A
        int    a;       4 < 8      4        8      [8, 11] 
        double b;       8 = 8      8        16     [16, 23]
        float  c;       4 < 8      4        24     [24, 27]
    };
        // 整体对齐系数 = min((max(int,double ,float), 8) = 8
        // 将内存大小由28补齐到8的整数倍32,∴ result = 32
};

练习

1.

struct x_ {
    char  a;      // 1 byte
    int   b;      // 4 bytes
    short c;      // 2 bytes
    char  d;      // 1 byte
} MyStruct1;

struct y_ {
    int   b;      // 4 bytes
    char  a;      // 1 byte
    char  d;      // 1 byte
    short c;      // 2 bytes
} MyStruct2;

NSLog(@"%lu---%lu--", sizeof(MyStruct1), sizeof(MyStruct2));
//12---8--

2.

struct StructOne {
    char   a;      // 1字节
    double b;      // 8字节
    int    c;      // 4字节
    short  d;      // 2字节
} MyStruct1;

struct StructTwo {
    double b;       // 8字节
    char   a;       // 1字节
    short  d;       // 2字节
    int    c;       // 4字节
} MyStruct2;
NSLog(@"%lu---%lu--", sizeof(MyStruct1), sizeof(MyStruct2));
//24---16--

3.

struct DD
{                       
    double  a;         
    char    b;           
    short   c;       
    struct FF 
    {
        int   a1;         
        char  b1;         
        short c1;          
        char  d1;         
    };
    char d; 
}Test;
NSLog(@"%lu", sizeof(Test));
//24

4.

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