之前我们有写过结构体类型,除了结构体类型以外我们的构造类型还包括位段、枚举、联合体。
位是二进制位
位段的声明和结构是类似的,有两点不同
1.位段的成员分别是int ,unsigned int 或 signed int(char)
2.位段的成员名后边有一个冒号和一个数字
举例
- struct S
- {
- int a : 2;
- int b : 5;
- int c : 10;
- int d : 30;
- };
这表示a占2个比特位,b占5个比特位,c占10个比特位,d占30个比特位。
所以我们在主函数内求sizeof(S)时结果是8,8代表8个字节,1字节=8比特位
那么你就会纳闷了,2+5+10+30 = 47,怎么能是64比特位呢?请看下面的解释
👇从内存的角度看位段
当编译器看到int的时候,会直接开辟4个字节(32比特位)的空间。
好,a占了2比特位,b占了5比特位,c占了10比特位。还剩15比特位,不够放d呀,那就只能再开辟32比特位空间来放d。
所以:①结果是64比特位②位段冒号后面的数字不能超过32,32本身也不行。
位段可以用来节省空间。
枚举就是一一列举,把可能的取值一一列举。
枚举的定义
enum 枚举名{枚举值列表};
举例:
- #include
- enum Sex
- {
- MALE ,
- FEMALE ,
- SECRET
- };
-
- int main()
- {
- enum Sex s = MALE; //枚举变量的定义
- printf("%d", s);
- return 0;
- }
枚举里是常量,比如MALE就是0,FEMALE就是1,SECRET就是2。所以main函数里打印出来的s值是0。
那你可能会问,我要是想定义其他常量数值我难道要一直写下去吗?
当然不用,比如你想用常量值5,那么你可以MALE = 5;再打印就是5了。
我们之前可以用#define定义常量,为什么非要用枚举?
枚举的优点:
1.增加代码可读性和可维护性
2.和#define定义的标识符比较,枚举有类型检查,更加严谨
3.防止了命名污染(封装)
4.便于调试
5.使用方便,可以一次性定义多个常量
一种特殊的自定义类型,这种类型定义的变量也包含一系列的成员,特征是这些成员共用同一块空间(所以联合也叫共用体)
举例
- #include
- union Un
- {
- char c;
- int i;
- };
-
- int main()
- {
- union Un u;
-
- printf("%d\n", sizeof(u)); //4
- return 0;
- }
共用体所占4字节空间,且共用体及其中的c,i的地址也都一样。
联合体的特点:大小至少是最大成员的大小(因为它至少得有能力保存最大的那个成员),但同一时刻共用体成员只能用一个。
当最大成员大小不是最大对齐数的整数倍时,就要对齐到最大对齐数的整数倍。
数组按一个元素算。