目录
位段
什么是位段?
位段的声明和结构是类似的,有两个不同:
1.位段的成员必须是 int、unsigned int 或signed int
2.位段的成员名后边有一个冒号和一个数字
举个例子
struct A
{
int _a:2;
int _b:5;
int _c:10;
int _d:30;
};//注冒号后数字表示变量占多少比特位
位段的内存分配
1. 位段的成员可以是 int ,unsigned int ,signed int 或者是 char 类型
2. 位段的空间上是按照需要以4个字节( int )或者1个字节( char )的方式来开辟的
3. 位段涉及很多不确定因素,位段是不跨平台的,注重可移植的程序应该避免使用位段
对比一下和结构体的区别
可以看到通过位段的使用可以减少空间
struct A
{
//4byte
int _a:2;
int _b:5;
int _c:10;
//4byte
int _d:30;
};
我们想法对嘛我们实验一下
位段跨平台问题
1. int 位段被当成有符号数还是无符号数是不确定的
2. 位段中最大位的数目不能确定。(16位机器最大16,32位机器最大32,写成27,在16位机
器会出问题
3. 位段中的成员在内存中从左向右分配,还是从右向左分配标准未定义
4. 当一个结构包含两个位段,第二个位段成员比较大,无法容纳于第一个位段剩余的位时,是
舍弃剩余的位还是利用,这是不确定的
总结
跟结构相比,位段可以达到同样的效果,但是可以很好的节省空间,但是有跨平台的问题存在
枚举
枚举特点
枚举顾名思义就是一一列举,把可能的取值一一列举
比如现实生活中的:星期,月份,性别等等都会运用到枚举
enum Day//星期
{
Mon,
Tues,
Wed,
Thur,
Fri,
Sat,
Sun
};
enum为枚举关键词,enum Day为枚举类型
{}中内容为枚举的可能取值叫做“枚举常量”
这些可能取值都是有值的,默认从0开始,一次递增1,当然在定义的时候也可以赋初值
enum Color
{
RED=1,
GREEN=2,
BLUE=4
}
枚举优点
1. 增加代码的可读性和可维护性
2. 和#define定义的标识符比较枚举有类型检查,更加严谨
3. 防止了命名污染(封装)
4. 便于调试
5. 使用方便,一次可以定义多个常量
联合体(共用体)
联合也是一种特殊的自定义类型,这种类型定义的变量也包含一系列的成员,特征是这些成员公用同一块空间(所以联合也叫共用体)
举个例子
union un
{
char c;
int i;
};
int main()
{
union un u;
printf("%d\n", sizeof(u));
printf("%p\n", &(u.c));
printf("%p\n", &(u.i));
return 0;
}
可以看到我们定义了一个联合体变量u 内部包含char类型和int类型变量这时联合体的大小是多少呢以及变量间地址差是多少呢,我们来实验一下
看到结果我们知道了当前联合体大小为int类型大小并且内部变量的起始地址是相同的
联合体特点
联合的成员是共用同一块内存空间的,这样一个联合变量的大小,至少是最大成员的大小
判断当前计算机的大小端存储
知道了联合体的特点我们就可以试试来写一道面试题
//int check_sys()
//{
// int a = 1;
// return (*(char*)&a);
//}
int check_sys()
{
union
{
int i;
char c;
}u;
u.i = 1;
return u.c;
}
int main()
{
int ret=check_sys();
if (ret == 1)
{
printf("小端\n");
}
else
{
printf("大端\n");
}
return 0;
}
联合体大小计算
结论: 联合的大小至少是最大成员的大小
当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍
以上就是我对于位段+枚举+联合体的总结 觉得有用的还请点赞评论!!!蟹蟹了