2025.4.27_C_Struct,Enum,Union
1.结构体Struct
1.1 概念
一系列具有相同类型或不同类型的数据构成的数据集合。
struct person
{char* name;uint8_t gender;
}p1,p2; //p1,p2是struct person类型的变量
1.2 匿名结构体类型
struct
{char* name;uint8_t gender;
};
这时结构体对象只能在声明类型的时候创建变量,其余时候因为没有名字,所以不能创建变量。
1.3 结构体的自引用
结构体的成员不能是自身,因为无法确定大小,但是可以是自身类型的指针(链表,树)。
1.4 typedef + 结构体
typedef struct person
{char* name;uint8_t gender;
}person;
这种写法定义了一个struct person类型,并且给他定义了一个别名person,在创建对象的时候既可以使用struct person也可以直接使用person。
typedef struct
{char* name;uint8_t gender;
}person;
这种写法定义了一个匿名结构体类型,并且给他指定了别名person,在创建对象时只能使用person,不能使用struct person,因为匿名结构体对象除了在声明的时候创建对象外其余时候都不能创建对象。
1.5 结构体所占内存大小
1.5.1结构体的对齐规则:
1.第一个成员在与结构体变量偏移量为0的地址处。
2.其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。
对齐数=编译器默认的一个对齐数 与 该成员大小的较小值。
3.结构体总大小为最大对齐数(每个成员变量都有一个对齐数)的整数倍。
4.如果嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍
。
例子1:
struct s2
{char c1;char c2;int i;
};
printf("%d\n", sizeof(struct s2)); // 输出8
为什么会输出8?因为在按照规则2,存储b的时候他们需要对齐到char类型大小的的整数倍地址处,也就是要对齐1的整数倍,所以a存储在偏移量为0的地址,b存储在偏移量为1的地址;但是int c,按照规则,它需要对齐到4的整数倍的地址处,所以它的地址偏移量为4,中间偏移量为2和3的地址为空闲状态。所以这个结构体的整体大小为8。
例子2:
struct s1
{char c1;int i;char c2;
};
printf("%d\n", sizeof(struct s1)); // 输出12
为什么输出12?因为i是int类型,它的地址需要对齐4的整数倍,所以它的地址偏移量为4,c2只需要对齐1的整数倍就可以,所以目前占用地址数是9,但是根据
规则3:结构体总大小为最大对齐数(每个成员变量都有一个对齐数)的整数倍
所以总大小需要对齐结构体成员中最大对齐数的整数倍(这里是4),但是9不满足,所以必须对齐到12.
例子3:
struct s3
{double d;char c;int i;
};
printf("%d\n", sizeof(struct s3)); // 输出16
为什么输出16?因为成员d占用八个地址,c只需要对齐1的整数倍,所以它在偏移量为8的地址处,i需要对齐4的整数倍,明显9不满足,所以它存储在偏移量为12的地址处。所以目前s3占用16个字节,根据规则三:要对其成员中最大大小的整数倍,这里刚好是16,所以满足。
例子4:
struct s4
{char c1;struct s3 s;double d;
};
printf("%d\n", sizeof(struct s4)); // 输出32
为什么输出32?c1在偏移量为0的地址处,s占用16个字节,但是根据规则2:
2.其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。
对齐数=编译器默认的一个对齐数 与 该成员大小的较小值。
其中,VS的默认对齐数是8,所以s只需要对齐偏移量为8的地址处,所以它在8-24的地址处;d是double类型,它需要对齐8的整数倍偏移量,刚好24是8的倍数,不用空闲地址,所以目前一共占用32个字节,根据规则4:
4.如果嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍
这里因为存储s的时候对齐的是VS默认的对齐数8,所以这里struct s4 占用的总空间只需要对齐8的倍数,而不是16。刚好32满足条件,所以struct s4的大小是32.
1.5.2 为什么存在内存对齐?
1.5.3 怎么修改编译器的默认字节对齐数?
#program pack(n)
...
#program pack()
例子:
#pragma pack(1)
struct s3
{char c1;char c2;int i;
};
#pragma pack()
现在结构体s3中的编译器默认字节对齐数变成了1, 所以i只需要对齐1的整数倍,整个s3也只需要对齐1的整数倍偏移量,所以s3占用的字节数是6。
1.5.4 offsetof() 查看成员的结构体偏移量
struct s4
{char c1;struct s3 s;double d;
};int main()
{printf("%d", offsetof(struct s4 , d)); //输出16
}