结构体联合体的字节对齐问题详解:
1.字节对齐的原因解释
比如有些平台每次读都是从偶地址开始,如果一个int型(假设为32位系统)
如果存放在偶地址开始的地方,那么一个读周期就可以读出这32bit,而如果存
放在奇地址开始的地方,就需要2个读周期,并对两次读出的结果的高低字节进
行拼凑才能得到该32bit数据。
为了提高效率,就有了字节对齐。
2.字节对齐的基本操作方式:
1) 结构体变量的首地址能够被其最宽基本类型成员的大小所整除;
2) 结构体每个成员相对于结构体首地址的偏移量都是成员大小的整数倍,如有需
要编译器会在成员之间加上填充字节;例如上面第二个结构体变量的地址空间。
3) 结构体的总大小为结构体最宽基本类型成员大小的整数倍,如有需要编译器会
在最末一个成员之后加上填充字节。
3.各种实例分析理解(最好从事实例子出发去理解),特别是对第三个例子的理解。
(包含的类型相同,但排列顺序不同也会有不同的结果)
1)typedef struct unRec
{
int ullndex;
int usLevel;
short ucPos;
}REC_S;
REC_S stMax, *pstMax;
int d = sizeof(stMax);
printf("%d",d);
在这种情况下,结构体中占空间最大的类型是int型(4字节),根据字节对齐原
理,short(2字节)后补两字节空间,输出为12;
2)typedef struct unRec
{
char ar;
int usLevel;
char er;
}REC_S;
REC_S stMax, *pstMax;
int d = sizeof(stMax);
printf("%d",d);
在这种情况下,结构体中占空间最大的类型是int型(4字节),根据字节对齐原
理,第一个char(1字节)后补三字节空间,第二个char(1字节)后补三字节空间,
输出为12;
3)typedef struct unRec
{
char ar;
char er;
int usLevel;
}REC_S;
REC_S stMax, *pstMax;
int d = sizeof(stMax);
printf("%d",d);
在这种情况下,结构体中占空间最大的类型是int型(4字节),根据字节对齐原
理,第一个char和第二个char连续排列占两字节,之后补两字节,输出为8;
4)typedef struct unRec
{
char ar[11];
int usLevel;
char er;
}REC_S;
REC_S stMax, *pstMax;
int d = sizeof(stMax);
printf("%d",d);
在这种情况下,结构体中占空间最大的类型是int型(4字节),其他均为char类型,
包括char的数组在内存中也是以基本的char存放。这时候由于char数组和之后的单个
char被int分开,在内存中不可连续,各自完成各自字节对齐。char ar[11]补一个字节
对齐,占12个字节,为int型(4字节)整数3倍,最后的单个char补三字节凑成4字节为
int型一倍,最后输出为20;
1.字节对齐的原因解释
比如有些平台每次读都是从偶地址开始,如果一个int型(假设为32位系统)
如果存放在偶地址开始的地方,那么一个读周期就可以读出这32bit,而如果存
放在奇地址开始的地方,就需要2个读周期,并对两次读出的结果的高低字节进
行拼凑才能得到该32bit数据。
为了提高效率,就有了字节对齐。
2.字节对齐的基本操作方式:
1) 结构体变量的首地址能够被其最宽基本类型成员的大小所整除;
2) 结构体每个成员相对于结构体首地址的偏移量都是成员大小的整数倍,如有需
要编译器会在成员之间加上填充字节;例如上面第二个结构体变量的地址空间。
3) 结构体的总大小为结构体最宽基本类型成员大小的整数倍,如有需要编译器会
在最末一个成员之后加上填充字节。
3.各种实例分析理解(最好从事实例子出发去理解),特别是对第三个例子的理解。
(包含的类型相同,但排列顺序不同也会有不同的结果)
1)typedef struct unRec
{
int ullndex;
int usLevel;
short ucPos;
}REC_S;
REC_S stMax, *pstMax;
int d = sizeof(stMax);
printf("%d",d);
在这种情况下,结构体中占空间最大的类型是int型(4字节),根据字节对齐原
理,short(2字节)后补两字节空间,输出为12;
2)typedef struct unRec
{
char ar;
int usLevel;
char er;
}REC_S;
REC_S stMax, *pstMax;
int d = sizeof(stMax);
printf("%d",d);
在这种情况下,结构体中占空间最大的类型是int型(4字节),根据字节对齐原
理,第一个char(1字节)后补三字节空间,第二个char(1字节)后补三字节空间,
输出为12;
3)typedef struct unRec
{
char ar;
char er;
int usLevel;
}REC_S;
REC_S stMax, *pstMax;
int d = sizeof(stMax);
printf("%d",d);
在这种情况下,结构体中占空间最大的类型是int型(4字节),根据字节对齐原
理,第一个char和第二个char连续排列占两字节,之后补两字节,输出为8;
4)typedef struct unRec
{
char ar[11];
int usLevel;
char er;
}REC_S;
REC_S stMax, *pstMax;
int d = sizeof(stMax);
printf("%d",d);
在这种情况下,结构体中占空间最大的类型是int型(4字节),其他均为char类型,
包括char的数组在内存中也是以基本的char存放。这时候由于char数组和之后的单个
char被int分开,在内存中不可连续,各自完成各自字节对齐。char ar[11]补一个字节
对齐,占12个字节,为int型(4字节)整数3倍,最后的单个char补三字节凑成4字节为
int型一倍,最后输出为20;