二、位段
2.1什么是位段
位段的声明和结构是类似的,有两个不同:
1.位段的成员必须是 int、unsigned int 或signed int 。
2.位段的成员名后边有一个冒号和一个数字。
例如:
struct A { int a:2; int b:5; int c:10; int d:30; };
A就是一个位段类型。 那位段A的大小是多少
4个整型应该是16个字节,现在为什么是8个呢?
在内存中有些数据很小,例如存储1,2,3只需要两个二进制位,使用位段就可以节省30个比特位。
位段是站在节省空间的角度考虑的
2.2位段的内存分配
注意:
1. 位段的成员可以是 int unsigned int signed int 或者是 char (属于整形家族)类型
2. 位段的空间上是按照需要以4个字节( int )或者1个字节( char )的方式来开辟的。
3. 位段涉及很多不确定因素,位段是不跨平台的,注重可移植的程序应该避免使用位段。
struct S { char a : 3; char b : 4; char c : 5; char d : 4; }; int main() { struct S s = { 0 }; s.a = 10; s.b = 12; s.c = 3; s.d = 4; return 0; }
2.3位段的跨平台问题
1. int 位段被当成有符号数还是无符号数是不确定的。
2. 位段中最大位的数目不能确定。(16位机器最大16,32位机器最大32,写成27,在16位机 器会出问题。
3. 位段中的成员在内存中从左向右分配,还是从右向左分配标准尚未定义。
4. 当一个结构包含两个位段,第二个位段成员比较大,无法容纳于第一个位段剩余的位时,是 舍弃剩余的位还是利用,这是不确定的。
总结: 跟结构相比,位段可以达到同样的效果,并且可以很好的节省空间,但是有跨平台的问题存在。
三、枚举
枚举顾名思义就是一一列举。
把可能的取值一一列举。
比如我们现实生活中:
一周的星期一到星期日是有限的7天,可以一一列举。
性别有:男、女、保密,也可以一一列举。
三原色,也可以一一列举。
3.1枚举类型的定义
enum Day//星期 { Mon, Tues, Wed, Thur, Fri, Sat, Sun }; enum Sex//性别 { MALE, FEMALE, SECRET }; enum Color//颜色 { RED, GREEN, BLUE };
以上定义的 enum Day , enum Sex , enum Color 都是枚举类型。
{}中的内容是枚举类型的可能取值,也叫枚举常量 。
枚举常量的值默认从0开始,依次递增1
在声明枚举类型的时候也可以赋初值
3.2枚举的使用
enum Color { RED=1, GREEN=2, BLUE=4 }; enum Color clr = GREEN;//只能拿枚举常量给枚举变量赋值,才不会出现类型的差异。
enum Color clr = 3;
这段代码是错误的,3是int类型,clr变量是Color类型 ,无法赋值成功。说明枚举是一种类型
3.3枚举的优点
我们可以使用 #define 定义常量,为什么非要使用枚举?
枚举的优点:
1. 增加代码的可读性和可维护性
2. 和#define定义的标识符比较枚举有类型检查,更加严谨。
3. 便于调试
4. 使用方便,一次可以定义多个常量
四、联合(共用体)
4.1联合体的定义
联合也是一种特殊的自定义类型
这种类型定义的变量也包含一系列的成员,特征是这些成员公用同一块空间(所以联合也叫共用体)。
例如:
union Un { int i; char a; }; int main() { union Un un = { 0 }; un.i = 0x11223344; un.a = 0x55; return 0; }
4.2联合体的特点
联合的成员是共用同一块内存空间的,这样一个联合变量的大小,至少是最大成员的大小(因为联 合至少得有能力保存最大的那个成员)。
union Un { int i; char a; }; int main() { union Un un = { 0 }; printf("%p\n", &un); printf("%p\n", &(un.i)); printf("%p\n", &(un.a)); return 0; }
例:判断当前计算机的大小端存储
union { int i; char a; }un; int check() { union { int i; char a; }un = { .i = 1 }; return un.a; } int main() { int ret = check(); if (ret == 1) { printf("小端\n"); } else { printf("大端\n"); } return 0; }
4.3联合大小的计算
联合的大小至少是最大成员的大小。
当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍。
本次的内容到这里就结束啦。希望大家阅读完可以有所收获,同时也感谢各位读者的支持。文章有问题可以在评论区留言,博主一定认真认真修改,以后写出更好的文章。