C语言结构体的内存对齐
如何计算该结构体的大小?
首先得掌握结构体的对齐原则。
1.第一个成员在于结构体变量偏移量为0的地址处
2.其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处
对齐数 编译器默认的一个对齐数 与 该成员大小的较小值。注:(Linux-gcc没有默认对齐数,vs的默认对齐数为8)
3.结构体总大小为最大对齐数(每个成员变量都有一个对齐数)的整数倍
4.如果是嵌套了结构体的情况,嵌套的结构体对齐到自己的最大整数倍处,结构体的整体大小就是所有最大对齐数(包含嵌套结构体的对齐数)的整数倍。
struct S1
{
char c1;
int i;
char c2;
};
struct S2
{
char c1;
char c2;
int a;
};
int main()
{
struct S1 s1 ={0};
printf("%d\n", sizeof(s1));//?
struct S2 s2 ={0};
printf("%d\n",sizeof(s2));//?
}
上图解读:
**第一步:**首先存储char c1,在内存中“随便”找一个位置,占内存大小1(红色区域),根据对齐原则一,此处位置为0。
**第二步:**接着存储int i,计算其对齐数,其内存大小4字节,vs默认对齐数8,取较小值,所以int i的对齐数为4.根据结构体对齐原则二,其存放的位置应为整数倍,也就是说,它只能存放在4、8、12…这些位置,所以这里把它存到4位置,占4个字节,如紫色区域。
**第三步:**最后存储char c2,计算其对齐数为1,根据对齐原则二,其可存储的位置为1,2,3,4…,所以这里把c1接着存储到9位置。这还没完,现在s1大小才为9
**第四步:**根据对齐原则三,s1的大小必须为4的倍数,所以取s1大小为12.
同理,咱们还可以得到结构体s2的大小
可见结构体的大小与成员变量的顺序有一定关系,且**空间小的变量放在一起(比如这里的char放在一起)**可减小结构体空间使用
嵌套结构体情况计算大小
#include<stdio.h>
struct S3
{
double d;
char c;
int i;
}s3;
struct S4
{
char c1;
struct S3 s3;
double g;
}s4;
int main()
{
printf("%d\n", sizeof(s4));
}
问:s4的大小?
上图解读:
**第一步:**首先存储char c1,在内存中“随便”找一个位置,占内存大小1(红色区域),根据对齐原则一,此处位置为0。
**第二步:**接着存储S3,S3里面又有d、c、i,所以依次存储d、c、i,计算d对齐数,其内存大小8字节,vs默认对齐数8,取较小值,所以double d的对齐数为8。根据结构体对齐原则二,其存放的位置应为整数倍,也就是说,它只能存放在8、16、24…这些位置,所以这里把它存到8位置,占8个字节,如紫色区域。再存储char c,对齐数为1,直接存储在位置16,占1个字节,再接着存储int i,i的对齐数为4,所以把它存到20位置。
**第三步:**最后存储double g,计算其对齐数为8,根据对齐原则二,把g存储到24位置。这还没完,现在s1大小为32
**第四步:**根据对齐原则三,s1的大小必须为8的倍数,所以24满足要求,所以最终的S4大小为32。