#字节对齐

[TOC]

什么是字节对齐(可以跳过)

现代计算机中内存空间都是按照字节(byte)划分的,从理论上讲似乎对任何类型的变量的访问可以从任何地址开始,但实际情况是在访问特定变量的时候经常在特定的内存地址访问,这就需要各类型数据按照一定的规则在空间上排列,而不是顺序地一个接一个地排放,这就是对齐.

字节对齐的好处(可以跳过)

为了提高效率,计算机从内存中取数据是按照一个固定长度的。以32位机为例,它每次取32个位,也就是4个字节(每字节8个位)。字节对齐有什么好处?以int型数据为例,如果它在内存中存放的位置按4字节对齐,也就是说1个int的数据全部落在计算机一次取数的区间内,那么只需要取一次就可以了.

如何对齐

通常,我们写程序的时候,不需要考虑对齐问题,编译器会替我们选择适合目标平台的对齐策略。但是,正因为我们一般不需要关心这个问题,所以,如果编辑器对数据存放做了对齐,而我们不了解的话,常常会对一些问题感到迷惑。最常见的就是struct数据结构的sizeof结果,例如:

typedef struct
{
    char  member1;
    short member2;
    int   member3;
}Family;

//打印长度
NSLog(@"Family size is %zd",sizeof(Family));

//输出结果
2016-07-22 15:34:29.081 Study[14587:5575156] Family size is 8

//下面我们更换一下成员变量位置,看看有什么效果

typedef struct
{
    char  member1;
    int   member3;
    short member2;
}Family;

//打印长度
NSLog(@"Family size is %zd",sizeof(Family));

//输出结果

2016-07-22 15:36:25.126 Study[14591:5575689] Family size is 12

那么问题来了,两个结构体的成员变量只是改变了下顺序,为什么占用的内存大小不同呢?

对齐原则:

  • char 偏移量必须为sizeof(char) 即1的倍数,可以任意地址开始存储
  • short 偏移量必须为sizeof(short) 即2的倍数,只能从0,2,4...等2的倍数的地址开始存储
  • int 偏移量必须为sizeof(int) 即4的倍数,只能从0,4,8...等4的倍数的地址开始存储
  • float 偏移量必须为sizeof(float) 即4的倍数,只能从0,4,8...等4的倍数的地址开始存储
  • double 偏移量必须为sizeof(double)即8的倍数,只能从0,8,16...等地址开始存储

根据以上原则,我们来分析:

typedef struct
{
    char  member1;
    short member2;
    int   member3;
}Family;

  • 这个结构体:member1占一个字节,即 存储位置为 0
  • member2占两个字节,根据上面原则,开始存储地址应该是2的倍数,即 2~3
  • member3占4个字节,根据原则,开始存储地址是4的倍数,即 4~7
  • 总共占用了0 ~ 7 共8个字节.
    内存存储图为:


    Paste_Image.png

下面分析:

typedef struct
{
    char  member1;
    int   member3;
    short member2;
}Family;

  • 这个结构体:member1占一个字节,即 0
  • member2占4个字节,根据上面原则,开始存储地址应该是4的倍数,即 4~7
  • member3占2个字节,根据上面原则,开始存储地址是2的倍数,即 8 ~ 9
    总共占用了0 ~ 9 应该是10个字节,但为什么实际却是12个字节呢 ?
    因为默认对齐方式是4字节(至于为什么,往下看),也就是说,总长度必须是4的倍数,因此长度既要大于10,还要是4的倍数,那就是12了.
    内存结构图为:


    Paste_Image.png

结构体如何设定字节对齐

  • 当未明确指定时,以结构体中最长的成员的长度为其有效值,上面的两个结构体都是int类型最长,也就是4字节对齐

  • 结构体作为成员:如果一个结构里有某些结构体成员,则结构体成员要从其内部最大元素大小的整数倍地址开始存储.(struct a里存有struct b,b里有char,int ,double等元素,那b应该从8的整数倍开始存储.)

  • 结构体的总大小,也就是sizeof的结果,必须是其内部最大成员的整数倍.不足的要补齐.

  • 当用#pragma pack(n)指定时,以n和结构体中最长的成员的长度中较小者为其值.

  • 用#pragma pack()为还原字节对齐为默认值.

  • attribute ((packed)) 1字节对齐,此时结构体的长度就是各成员变量长度之和.

例如:

pragma pack 的用法

struct A {
    int   a;
    char  b;
    short c;
};

struct B {
    char  b;
    int   a;
    short c;
};

#pragma pack (2) /*指定按2字节对齐*/
struct C {
    char  b;
    int   a;
    short c;
};
#pragma pack () /*取消指定对齐,恢复缺省对齐*/



#pragma pack (1) /*指定按1字节对齐*/
struct D {
    char  b;
    int   a;
    short c;
};
#pragma pack ()/*取消指定对齐,恢复缺省对齐*/

//计算所占字节
int s1=sizeof(struct A);
    int s2=sizeof(struct B);
    int s3=sizeof(struct C);
    int s4=sizeof(struct D);
    
    printf("%d\n",s1);
    printf("%d\n",s2);
    printf("%d\n",s3);
    printf("%d\n",s4);
//输出结果:
8
12
8
7

attribute ((packed))的用法:
让指定的结构结构体按照 1 字节对齐,例如:

//不加packed修饰
typedef struct {
    char    version;
    int16_t sid;
    int32_t len;
    int64_t time;
} Header;

//计算长度
NSLog(@"size is %zd",sizeof(Header));
输出结果为:
2016-07-22 11:53:47.728 Study[14378:5523450] size is 16

可以看出,默认系统是按照4字节对齐

//加packed修饰
typedef struct {
    char    version;
    int16_t sid;
    int32_t len;
    int64_t time;
}__attribute__ ((packed)) Header;

//计算长度
NSLog(@"size is %zd",sizeof(Header));
输出结果为:
2016-07-22 11:57:46.970 Study[14382:5524502] size is 15

用packed修饰后,变为1字节对齐,这个常用于与协议有关的网络传输中.

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

推荐阅读更多精彩内容

  • @[c++|struct] 今天在编程中碰到一个坑,搞的调试了半天,最后发现程序中在写数据和读取数据时结构体定义不...
    drybeans阅读 3,595评论 1 11
  • 4.3:新增class的相关内容 今天看到一个题目: 最开始简单的理解为,每个数据的size之和就是偏移量。因为偏...
    AwesomeAshe阅读 733评论 0 0
  • 1. 对齐 需要各类型数据按照一定的规则在内存空间上排列,而不是顺序的排放,这就是对齐。 2. 对齐的原因 最常见...
    Sheldor936阅读 368评论 0 0
  • (字节对齐的实现细节和编译器有关) 1. 基本概念 字节对齐:计算机存储系统中以Byte为单位存储数据,不同数据类...
    安然_fc00阅读 2,876评论 0 1
  • 通过一段代码来描述内存对齐的现象。 上述代码打印出来的结果为:24,16 为什么相同的结构体,只是交换了变量 ab...
    豆瓣菜阅读 6,717评论 5 26