提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档
共用体
假设一个场景,电脑储存字节数大于 1 的数据都涉及了大小端储存
复习一下大小端储存
int i = 0x11223344;
在内存中存放
如果是小端,低字节位放在低地址,高字节放在高地址
如果是大端,低字节位放在高地址,高字节放在低地址
我们可以怎么样来判断这个电脑是大小端储存呢?
如果我们把储存的最低地址的值拿出来,是不是取出的值不一样,按照这个不一样,就能够判断机器的大小端了
这里介绍一个类型,可以很方便地实现这个功能
共用体的创建和使用
这个类型就是共用体,或者叫联合
创建一个类型
union Un
{
char a;
int i;
double c;
};
共用体,顾名思义就是 几个数据用的是同一个空间
那么他就有几个特性
- 如果我修改任意的共用体成员,我会把所有的成员的值都改了
- 不同的成员如果类型不同,访问的空间大小是不一样的,我可以利用他们的不同空间来切分共用体的整块空间
利用第二个特性,我们就能够完成大小端的判断
union Un
{
char c;
int i;
};
int main()
{
union Un a;
a.i = 1;//使用和 结构体相似
printf("%d",a.c);
return 0;
}
我先定义了一个共用体,两个成员共用一块内存,一个成员是整形,一个是字符类型
然后把整形的类型赋值为1(使用也是用 变量.成员 的方式使用)
那在小端机器中,数据值这样放的
如果我们把 a.c 拿出来就是红框的数据 也就是 1
但是如果在大端机器中则是
如果把char类型的数据拿出来,就是 0
这样就能分别大小端存储
同时也利用了切分的性质
共用体的大小
一个类型总是有大小的
首先我先猜一下,因为是多个成员共用一个内存,是不是就意味着最大的成员能够放进去,所以共用体大小至少是最大的成员的大小
但是这就够了吗?
不够
我们看看这个代码
#define _CRT_SECURE_NO_WARNINGS 1
union Un
{
short a[3];//长度是6,对齐数是2
int i;//长度是4,对齐数是4
};
//如果刚好够 是不是 长度为 6 就可以了?
int main()
{
printf("%d",sizeof(union Un));
return 0;
}
如果按照猜测,是不是 union Un 长度是 6 就可以了?
但是看看结果
长度是 8 !
怎么来的?
实际上,共用体的长度不仅仅和成员长度有关,还和成员的最大对齐数相关,他的长度必须是共用体最大对齐数的整数倍
看代码注释
union Un
{
short a[3];//长度是6,对齐数是2
int i;//长度是4,对齐数是4
};
// 4和2最大对齐数是 4 , 长度大于等于 6 ,最终长度是 2*4 = 8
4和2最大对齐数是 4 , 长度大于等于 6 ,最终长度是 2*4 = 8
所以共用体长度是
- 长度最少是最大成员的长度
- 长度是最大对齐数的整数倍
枚举
我们在表达的时候,有一种需求是要表示一种一连串的数据
比如
月份的十二个月
24节气
一周七天等等
我们就可以用枚举储存起来
我们可以创建一个枚举类型
enum Week
{
MonDay = 1,
Tues,
Wed,
Thur = 8,
Fri,
Sat,
Sun
};
当我们创建枚举时
其中的成员我们可以赋给初值
后面的成员的值就依次加一
我们把 Week 里面的值按顺序打印一下
如果没有赋初值,第一个成员默认从0开始,后面成员依次加1
但是,当我们创建完类型,里面的值就相当于常量无法改变
enum Week
{
MonDay = 1,
Tues,
Wed,
Thur = 8,
Fri,
Sat,
Sun
};
int main
{
Monday = 10;//直接修改 enum类型内是违规的
enum Week a = Monday;//这个是定义枚举变量
a = 10;//可以修改
int q = Monday + Fri;//枚举也可以直接进行运算
}
枚举因为在定义变量的时候,只能取enum内所有成员的一个成员,每个成员都是整形,所以长度就是一个 int 的长度 也就是 4
能看到这的老铁都是大佬,希望多多支持,及时反馈,谢谢大家