要判断一个结构体所占的空间大小,大体来说分三步走:
1.先确定实际对齐单位,其由以下三个因素决定
(1) CPU周期
WIN vs qt 默认8字节对齐
Linux 32位 默认4字节对齐,64位默认8字节对齐
(2) 结构体最大成员(基本数据类型变量)
(3) 预编译指令#pragma pack(n)手动设置 n--只能填1 2 4 8 16
**上面三者取最小的** , 就是实际对齐单位(这里的“实际对齐单位”是我为了方便区分随便取的概念)
2.除结构体的第一个成员外,其他所有的成员的地址相对于结构体地址(即它首个成员的地址)的偏移量必须为实际对齐单位或自身大小的整数倍(取两者中小的那个)
3.结构体的整体大小必须为实际对齐单位的整数倍。
知道了这个原则, 在计算时可以按照:
对齐原则:每一成员的结束偏移量需对齐为后一成员类型的倍数
补齐原则:最终大小补齐为成员中最大值的倍数
例如:
struct MyStruct
{
int i; // (0) + 4 = 4,后面一个为2,已经对齐
char c; // (4) + 1 = 5,后面一个为结构体,直接拆开看第一个,int=4,所以对齐为(4) + 4 = 8
struct InnerStruct
{
int i; // (8) + 4 = 12,下一个为4,已经对齐(学java注意了,32位下C语言的long为4,和java不一样)
long l; // (12) + 4 = 16,下一个为8,已经对齐
double d; // (16) + 8 = 24,下一个为1,已经对齐
char c; // (24) + 1 = 25,这是最后一个,将其对齐为最大值8的倍数,所以补齐为 (24) + 8 = 32,最后结果=32
} innerStruct;
};
在遇到嵌套的循环体时, 直接进入嵌套的循环体内部按顺序计算。
struct MyStruct
{
int i; // (0) + 4 = 4,后面一个为2,已经对齐
char c; // (4) + 1 = 5,后面一个为结构体,直接拆开看第一个,int=4,所以对齐为(4) + 4 = 8
struct InnerStruct
{
int i; // (8) + 4 = 12,下一个为4,已经对齐(学java注意了,32位下C语言的long为4,和java不一样)
long l; // (12) + 4 = 16,下一个为8,已经对齐
double d; // (16) + 8 = 24,下一个为1,已经对齐
char c; // (24) + 1 = 25,这是最后一个,将其对齐为最大值8的倍数,所以补齐为 (24) + 8 = 32,最后结果=32
} innerStruct;
};