结构体对齐和补齐
结构体的对齐和补齐的规则:
对齐:假定从零地址开始,每成员的起始地址编号,必须是它本身字节数的整数倍。
补齐:结构的总字节数必须是它最大成员的整数倍。
注意:在Linux系统下计算补齐、对齐时,成员超过4字节按4字节计算。
#include <stdio.h>
#pragma pack(4)//编译器按照4字节对齐
typedef struct data
{
char a[11];
int b;
char c;
double d;
short e;
}data;
int main()
{
int size = sizeof(data);
printf("%d\n",size);
//计算出40字节是因为系统是64位的,默认8字节对齐
//默认4字节对齐,结果是32
return 0;
}
1、首先第一个成员 char a[11],char类型是占一个字节的,所以第一个成员结束的地址范围是0~10,占了11个字节。 注意:地址的范围是假定从0开始的。
2、然后是第二个成员int b,int类型是占了4个字节的,然后第二个成员的起始地址是11,然后我们发现11不是4的倍数,所以这个时候我们要考虑对齐规则,即让起始地址是4的倍数,所以这里int b的起始地址是12,地址范围是12~15占了4个字节。
3、接着看第三个成员char c,因为char是占了一个字节,所以不需要考虑补齐和对齐,所以第三个成员占的地址是16,就一个字节。
4、再来看第四个成员double d,首先这里要注意一点在Linux系统下计算补齐、对齐时,成员超过4字节按4字节计算,这里我们默认是Linux系统下,当然在一般在别的系统下也有这个规则。然后这里的起始地址是17,不是4的倍数,所以我们要考虑对齐规则,即让double d的起始地址变成20,然后这里还要注意一点,double是4字节只是在对齐的时候需要考虑,在计算大小是还是要改成原来的8字节,所以第四个成员占的地址范围是20~27,占了8个字节。
5、最后再看第五个成员short e,short是占了2个字节的。起始地址是28,发现是2的倍数,所以不用再对齐了,所以占的地址范围是28~29,占了两个字节。
最后这个结构体的地址范围是0~29,大小是30个字节,但是我们在计算结构体大小的时候还有一条规则,就是补齐。在这个结构中最大的成员的字节数是4(超过4字节按4字节算),30不是4的整数倍,所以最后这个结构体的大小还要增加2,最后的结果就是32。
本文是复制大佬的文章。
原文链接:https://blog.csdn.net/weixin_45372436/article/details/99630449