iOS-Android-私房菜iOS技术专题

#字节对齐

2016-07-22  本文已影响997人  youngyunxing

[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

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

对齐原则:

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

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

下面分析:

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

结构体如何设定字节对齐

例如:

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字节对齐,这个常用于与协议有关的网络传输中.

上一篇下一篇

猜你喜欢

热点阅读