当前位置:网站首页>[Structural Internal Power Cultivation] The Mystery of Enumeration and Union (3)
[Structural Internal Power Cultivation] The Mystery of Enumeration and Union (3)
2022-08-05 07:55:00 【Albert Edison】
1. 枚举
枚举顾名思义就是一一列举.
把可能的取值一一列举.
比如我们现实生活中:
1、一周的星期一到星期日是有限的7天,可以一一列举.
2、性别有:男、女、保密,也可以一一列举.
3、月份有12个月,也可以一一列举
这里就可以使用枚举了.
枚举类型的定义
代码示例
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,当然在定义的时候也可以赋初值.
代码示例
#include <stdio.h>
enum Sex//性别
{
MALE,
FEMALE,
SECRET
};
int main()
{
printf("%d\n", MALE);
printf("%d\n", FEMALE);
printf("%d\n", SECRET);
return 0;
}
运行结果
枚举的优点
为什么使用枚举?
我们可以使用 #define
定义常量,为什么非要使用枚举?
#define MALE 4
#define FEMALE 5
#define SECRET 6
枚举的优点:
1、增加代码的可读性和可维护性;
2、和 #define
定义的标识符比较枚举有类型检查,更加严谨;
3、防止了命名污染(封装);
4、便于调试, #define
The defined constant cannot be debugged
5、使用方便,一次可以定义多个常量
枚举的使用
代码示例
#include <stdio.h>
enum Color//颜色
{
RED = 1,
GREEN = 2,
BLUE = 4
};
int main()
{
enum Color clr = GREEN;//只能拿枚举常量给枚举变量赋值,才不会出现类型的差异.
clr = 5;
return 0;
}
2. 联合(共用体)
联合类型的定义
联合也是一种特殊的自定义类型;
这种类型定义的变量也包含一系列的成员,特征是这些成员公用同一块空间(所以联合也叫 共用体).
代码示例
#include <stdio.h>
union Un
{
char c;
int i;
};
int main()
{
union Un u;
printf("%d\n", sizeof(u));
return 0;
}
那么 u 占多少个字节呢?我们打印看一下
为什么是 4 呢?
别急,我们把 u and its member addresses are all printed out and have a look
我的天啊,Why are all three of them the same address??
u、u.c、u.i they all point to the 1 first byte address;
c 占 1 个字节,i 占 4 个字节;
也就是 i 和 c shared the 1 个字节;
当我给 c 赋值的时候,我就把 i The first byte of the to change;
当我给 i 赋值的时候,整个 c also changed;
所以 联合体 比较特殊的地方在于:For members of the consortium at the same time can only use 1 个;
代码示例
#include <stdio.h>
union Un
{
char c;
int i;
};
int main()
{
union Un u = {
0 };
u.c = 'w';
u.i = 0x11223344;
return 0;
}
Let's take a look at the debug results in memory
首先把第 1 个字节改为 w
然后又把整个 i 的内容改为 0x1223344
联合的特点
1、联合的成员是共用同一块内存空间的,这样一个联合变量的大小,至少是最大成员的大小;
2、因为联合至少得有能力保存最大的那个成员;
联合的应用
For example, I'm going to design a school user system
1、学生:姓名、年龄、身份
2、老师:姓名、年龄、职称
代码示例
union type
{
identity; //身份
title; //职称
};
struct UserInfo
{
char name[20];
int age;
union type t;
};
身份 和 职称 are all similar values,But they will only use 1 个;
来看一道面试题:判断当前计算机的大小端存储
首先来看看什么是 大小端字节序
1、小端存储:假设 01 is the content of the lower byte,00 is the content of the high-order byte,put the contents of the lower byte 01 放在低地址处,高位字节的内容 00 放在高地址处,这种存储方式叫 小端存储
1、大端存储:假设 01 is the content of the lower byte,00 is the content of the high-order byte,put the contents of the lower byte 01 放在高地址处,高位字节的内容 00 放在低地址处,这种存储方式叫 大端存储
So how to do it with code?
其实很简单,we can come up with the 1 个字节的内容,如果第 1 个字节的内容是 1,那么就是 小端存储;如果第 1 个字节的内容是 0,那么就是 大端存储;
代码实现
#include <stdio.h>
int check_sys() {
int a = 1;
return (*(char*)&a);
}
int main()
{
int ret = check_sys();
if (1 == ret) {
printf("大端\n");
}
else {
printf("小端\n");
}
return 0;
}
运行结果
我们可以看到,a 的 int 类型,但我们在 check_sys()
实际使用的只有 1 个字节,Then update the code;
代码升级
#include <stdio.h>
int check_sys() {
union
{
char c;
int i;
}u;
u.i = 1;
return u.c;
}
int main()
{
int ret = check_sys();
if (1 == ret) {
printf("大端\n");
}
else {
printf("小端\n");
}
return 0;
}
运行结果
This is a clever use of 联合体,联合体 of members will share a space;
联合大小的计算
1、联合的大小至少是最大成员的大小.
2、当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍.
代码示例一
#include <stdio.h>
union Un1
{
char c[5];
int i;
};
int main()
{
printf("%d\n", sizeof(union Un1));
return 0;
}
成员 c 的类型是 char,So its alignment number is according to char 来算的,也就是 1,而 VS 默认的对齐数是 8,take the smaller value,Then its alignment number is 1;
成员 i 的类型是 int,So its alignment number is according to int 来算的,也就是 4,而 VS 默认的对齐数是 8,take the smaller value,Then its alignment number is 4;
成员 c 的对齐数是 1,成员 i 的对齐数是 4,Then the maximum number of alignments is 4,And the total size of the union must be an integer multiple of the maximum number of alignments;
又因为 c 是 char 类型的数组,有 5 个元素,所以有 5 个字节;i 是 int 类型,有 4 个字节;所以最大的就是 5 字节,But the maximum number of alignments is 4,不满足,So it is 8,也就是 4 的 2 倍.
运行结果
代码示例二
#include <stdio.h>
union Un2
{
short c[7];
int i;
};
int main()
{
printf("%d\n", sizeof(union Un2));
return 0;
}
c 是 short 类型,占 2 字节,有 7 个元素,所以占 14 字节;i 是int 类型,占 4 字节;
那么取 14 can be stored Un2 的所有成员了,但是 14 Is it an integer multiple of the maximum alignment number??
成员 c 的类型是 short,So its alignment number is according to short 来算的,也就是 2,而 VS 默认的对齐数是 8,take the smaller value,Then its alignment number is 2;
成员 i 的类型是 int,So its alignment number is according to int 来算的,也就是 4,而 VS 默认的对齐数是 8,take the smaller value,Then its alignment number is 4;
成员 c 的对齐数是 2,成员 i 的对齐数是 4,Then the maximum number of alignments is 4,And the total size of the union must be an integer multiple of the maximum number of alignments,14 不是 4 的倍数,怎么办呢?
很简单,浪费掉 2 字节变成 16 就是 4 的倍数了.
运行结果
边栏推荐
猜你喜欢
随机推荐
TRACE32——Break
【深度学习实践(一)】安装TensorFlow
Unity—物理引擎+“武器模块”
网络安全研究发现,P2E项目遭遇黑客攻击只是时间问题
不能比较或排序 text、ntext 和 image 数据类型
2006年星座运势全解-巨蟹
TRACE32——List源代码查看
TRACE32——外设寄存器查看与修改
DataFrame在指定位置插入行和列
TRACE32——Go.direct
Antdesign a-select 下拉框超出长度换行显示
browserslist 选项的目的是什么?
力扣刷题八月第一天
3555. 二叉树
Green Apple Forum reopens
线程池的创建及参数设置详解
YOLOv3 SPP理论详解(包括CIoU及Focal loss)
SVG big fish eat small fish animation js special effects
RedisTemplate: 报错template not initialized; call afterPropertiesSet() before using it
P1103 书本整理