为什么要对齐?
对于32 位字宽的计算机,每次访问内存的地址总是 4 的倍数。
为了尽量减少指令,提高数据处理性能
通常要求数据按字宽边界对齐,超过字宽依然对齐,不超过的则会补齐
对齐的原则
基本数据类型
如int,char,short,double为基本数据类型
取其字节为对齐字节
结构体
结构体对齐值为:
min{最大基本数据类型字节,指定对齐值N}
共用体
共用体中成员共用一块内存,内存选取成员中最大占用内存
共用体对齐值为min{指定值,最大基本数据类型字节}
预处理对齐值指令
# pragama pack(N)
//N为指定对齐值
# pragama pack(N)
//其中N一般选取1,2,4,8,16,32...
例子
#pragma pack(1)
//预处理N值为1
typedef struct {
char a[3];//1字节
int b;//4字节(最大)
short c;//2字节
} packed_struct;
//对齐值为1(比4小)
//sizeof(packed_struct):3+4+2=9
#pragma pack(2)
//预处理N值为2
typedef struct {
char a[3];//1字节
int b;//4字节(最大)
short c;//2字节
} packed_struct;
//对齐值为2(比4小)
//sizeof(packed_struct):4+4+2=10
//其中3字节被补充为对齐值2的整数倍(便于对齐)
#pragma pack(4)
//预处理N值为4
typedef struct {
char a[5];//1字节
} packed_struct;
//N=4,最大基本数据类型为char 1
//1<4
//所以对齐值为1
//5-->5
#pragma pack(2)
typedef union{
double a;//8字节
int b;//4字节
char c[9];//9字节
} union_struct;
//自身对齐值为9
//指定对齐值为2
//故对齐值为2
//size-->10(比9大的最小对齐值整数倍)
#pragma pack(4)
typedef union{
double a;//8字节
int b;//4字节
char c[9];//9字节
} union_struct;
//自身对齐值为9
//指定对齐值为4
//故对齐值为4
//size-->12(比9大的最小对齐值整数倍)
#pragma pack(4)
typedef union
{
char a[17];
//int b[2];
short c;
} p;
int main()
{
p s;
printf("%d", sizeof(s));
}
//N=4
//最大基本数据类型为short
//所以对齐值为2
//17-->18
//size-->18
#pragma pack(4)
typedef union
{
char a[17];
//int b[2];
//short c;
char b[3];
} p;
int main()
{
p s;
printf("%d", sizeof(s));
}
//N=4
//最大基本数据类型为char
//所以对齐值为1
//17-->17
//size-->17