深入C语言,发现多样的数据之枚举和联合体

一、枚举

枚举 是列出某些有穷序列集的所有成员的程序,或者是一种特定类型对象的计数。这两种类型经常(但不总是)重叠。是一个被命名的整型常数的集合。简单来说就将某种特定类型的对象一一进行列举,一一列举特定类型可能的取值

顾名思义就是一 一列举,把可能的取值一 一列举。

比如在我们的日常生活中:

每周的星期一到星期日都是有限的7天,可以一一列举
性别有:男、女,也可以一一列举。
月份有12个月,也可以一一列举

说白了,枚举常量属于枚举类型值是整形

1.1 枚举的定义

枚举的定义与结构体类似

enum 枚举类型名
{标识符1,标识符2,…
};
  • 枚举类型名受自己定义,如:week,year…,标识符就是其中的枚举常量,如Mon,Tues,Wed…
  • 每个枚举常量可以用一个标识符来表示,也可以为它们指定一个整数值,如果没有指定,那么默认从 0 开始递增。

1.2 枚举的使用

1.2.1 枚举的声明

如果我们想对一个星期,不用枚举用#define:

#define MON  1
#define TUE  2
#define WED  3
#define THU  4
#define FRI  5
#define SAT  6
#define SUN  7

使用枚举简化,代码示例如下:

enum DAY
{MON=1, //指定从1开始,否则默认从0开始TUE,WED,THU, FRI, SAT, SUN
};

与结构体类似,我们也可以使用typedef来简化枚举

typedef enum DAY
{MON,TUE,WED,THU,FRI,SAT,SUN
}DAY;

1.2.2 打印枚举常量

typedef enum DAY
{MON, TUE,WED,THU,FRI,SAT,SUN
}DAY;
int main()
{for (int i = MON; i < SUN; i++){printf("%d ", i);}return 0;
}

输出结果如下:

0 1 2 3 4 5 6 

 这也间接证明枚举是一个常量,默认从0开始,后面依次递增1。

1. 这些枚举类型的可能取值也叫做枚举常量,那既然是常量,就意味这不能被修改。

typedef enum DAY
{MON, TUE,WED,THU,FRI,SAT,SUN
}DAY;
int main()
{MON = 9;for (int i = MON; i < SUN; i++){printf("%d ", i);}return 0;
}

2. 虽然不能修改,但是我们可以在定义是给它们赋初值。

如果只给第一个赋初值,就从该初值开始,还是依次增1。

typedef enum DAY
{MON = 3, TUE,WED,THU,FRI,SAT,SUN
}DAY;
int main()
{for (int i = MON; i < SUN; i++){printf("%d ", i);}return 0;
}

输出结果为:

3 4 5 6 7 8 9

如果中间某个赋初值,前面的还是默认值,后面的会依次增1。

typedef enum DAY
{MON, TUE,WED,THU = 6,FRI,SAT,SUN
}DAY;
int main()
{for (int i = MON; i < SUN; i++){printf("%d ", i);}return 0;
}

输出结果为:

0 1 2 6 7 8 9

1.2.3 枚举变量的创建与初始化

那定义好了类型,我们就可以那这些类型来定义变量了:

typedef enum Color//颜色
{RED,GREEN,BLUE
}Color;int main()
{Color col = RED;
}

1.3 枚举的大小

已给枚举变量,只会用来存放一个枚举常量的值,所以一个枚举变量的大小应该就是一个 int 的大小,也就是4个字节。下面我们通过代码来测试一下。

#include <stdio.h>
enum color1
{RED,GREEN,BLUE
};enum color2
{GRAY = 0x112233445566,YELLOW,PURPLE
};int main()
{printf("%d\n", sizeof(enum color1));printf("%d\n", sizeof(enum color2));return 0;
}

输出结果为:

4

1.4 枚举的优点

于是这里就产生了一个问题,枚举类型的成员均为常量,不可在使用中被修改,那么我们同样可使用宏 #define 去定义常量,为什么非要使用枚举类型呢?

这是因为,相比于宏,枚举类型具有很多优点

  • 增加代码的可读性和可维护性
  • 和 #define 定义的标识符比较,枚举有类型检查,更加严谨
  • 防止了命名污染(封装)
  • 便于调试,#define在调试的时候会完成替换
  • 使用方便,一次可以定义多个变量

二、联合体

2.1 联合体的定义

和结构体一样联合也是一种特殊的自定义类型,这种类型定义的变量也包含有一系列的成员,但不同的是这些成员共用同一块空间(遂也被称作共用体)

union UN
{char c;int i;
};
//定义了一个共用体类型int main()
{union UN un;//定义了一个共用体变量printf("%d\n", sizeof(un));return 0;
}

2.2 联合体的特点

那现在大家来思考一个问题,上面的联合体变量un的大小是多少?

结果为4

这是为什么呢,在这里要注意上面的一句话:这些成员公用同一块空间(所以联合也叫共用体)

联合体un只有两个成员,char c; int i;c 大小1个字节,i是最大的成员4个字节。
那么因为它们共用同一块空间,所以四个字节就够了。
4个字节既可以放得下c ,也能放得下i。

联合的成员是共用同一块内存空间的,这样一个联合变量的大小,至少是最大成员的大小(因为联合至少得有能力保存最大的那个成员),受限于联合体的特点,我们不会同时使用联合里面的成员,只会用一种的一个。

我们要怎么证明这一点?

union UN
{char c;int i;
};
//定义了一个共用体类型int main()
{union UN un;//定义了一个共用体变量printf("%d\n", sizeof(un));printf("%p\n", &un);printf("%p\n", &un.c);printf("%p\n", &un.i);return 0;
}

输出结果为:

4
00EFFA2C
00EFFA2C
00EFFA2C

可以看出, un 、 un.c 、 un.i 它们三个的起始地址都一样,这也说明,联合体的成员确实会共用同一块内存空间。

那他们共用一块内存空间意味着什么呢?

是不是意味着它们不能同时存在啊,或者说它们不能同时使用。

就拿上面的哪个union Un来说:当我们使用成员 i 时,对于c来说,此时c的那一个字节里面是不是存的是 i 的内容,就相当于此时c是不存在的。

让我们来看一段代码:

	un.i = 0x11223344;un.c = 0x55;printf("%x\n", un.i);

因为它们共用一块空间,上述代码中先给un.i赋值。
那此时un的四个字节的空间里放的应该是这样的内容:

那我们在去给un.c赋值,是不是就会改变1个字节的内容(因为un.c是1个字节)。
应该就变成这样了:

那我们再以printf("%x\n", un.i);(%x是以16进制形式打印)打印出来是不是就是11223355了。
来验证一下:

输出结果为:

这样更好的验证了联合体的成员共用一块空间,不能同时存在。

2.3 联合体大小的计算

  1. 联合的大小至少是最大成员的大小
  2. 当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍。

来看下面这么一段代码:

#include<stdio.h>//联合体1:
union TEST1
{char c[5];int i;
};//联合体2:
union TEST2
{short c[7];int i;
};int main()
{//检查联合体的大小:printf("The size of TEST1 is %d\n", sizeof(union TEST1));printf("The size of TEST2 is %d\n", sizeof(union TEST2));return 0;
}

输出结果为:

The size of TEST1 is 8

The size of TEST2 is 16

在联合体 TEST1 中,占用空间最大的成员是 char 类型数组 c ,且其中含有 5 个元素,则其所占空间大小为 5 个字节,而我们都知道 VS 的对齐数默认为 8 ,则将会对齐至默认对齐数 4 的整数倍,即 8 个字节;而联合体 TEST2 中,占用空间最大的成员是 short 类型数组 c ,且其中含有 7 个元素,则其所占空间的大小为 14 个字节,那么就将会对齐至对齐数 4 的整数倍,即 16 个字节.

2.4 联合体判断机器大小端

我们早在学习数据在内存中的存储时就已经了解过一种判断大小端的方法,今天就为大家介绍另一种方法——通过联合体判断大小端。

那如何取出第一位呢?除了通过指针,我们也能利用联合体共用同一块内存这一性质判断。

代码如下:

int check_sys()
{union{int i;char c;}un;un.i = 1;return un.c; //返回1是⼩端,返回0是⼤端
}
int main()
{int ret = check_sys();if (ret == 1){printf("⼩端\n");}else{printf("⼤端\n");}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/1789.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

探索RadSystems:低代码开发的新选择(二)

系列文章目录 探索RadSystems&#xff1a;低代码开发的新选择&#xff08;一&#xff09;&#x1f6aa; 文章目录 系列文章目录前言一、RadSystems Studio是什么&#xff1f;二、用户认证三、系统角色许可四、用户记录管理五、时间戳记录总结 前言 在数字化时代&#xff0c;低…

【做一名健康的CSDNer】程序员哪几种行为最伤肾(程序员必看)

虽然没有专门针对程序员这一职业群体特有的伤肾行为的研究报道&#xff0c;但根据一般人群的健康风险和生活习惯&#xff0c;程序员由于其特殊的工作模式和环境&#xff0c;可能更容易出现如下伤肾的行为&#xff1a; 熬夜加班&#xff1a; 程序员由于项目进度、bug修复等原因&…

LOD2-Unity中Shader LOD技术原理以及使用

Shader LOD&#xff08;Level of Detail&#xff09;是Unity中用于优化渲染性能的一种技术。它通过在不同的距离或屏幕空间中使用不同的着色器来控制模型的细节级别&#xff0c;从而减少GPU的工作量。 Shader LOD的原理是根据相机与物体之间的距离来选择合适的着色器。在远离相…

高斯锁表导致sql报错处理

构造锁等待场景&#xff1a; 1.打开一个新的连接会话&#xff0c;使用普通用户连接GaussDB(DWS)数据库&#xff0c;在test SCHEMA 下创建测试表test.ypg_test。 CREATE TABLE ypg_test (id int, name varchar(50)); 2.开启事务1&#xff0c;进行INSERT操作。 START TRANSACTI…

【操作系统】——调度算法——小题

单核处理器系统中&#xff0c;有3个作业J1,J2,J3同时到达&#xff0c;其运行时间分别为1h&#xff0c;4h&#xff0c;2h&#xff0c;以单道方式运行&#xff0c;下面的执行序列中周转时间最短的是&#xff08;&#xff09; A,j1,j2,j3 Bj3,j2,j1 Cj2,j1,j3 Dj1,j3,j2 在各种作业…

GPFL个性化联邦学习:同时学习全局和个性化特征信息

Global category embedding指的是将全局类别信息嵌入到模型中的过程。在机器学习和深度学习中&#xff0c;当处理具有多个类别的数据集时&#xff0c;可以使用全局类别嵌入来将类别信息编码到模型中&#xff0c;以帮助模型更好地理解和利用类别之间的关系。这可以帮助提高模型在…

函数的创建和调用及删除

Oracle从入门到总裁:​​​​​​https://blog.csdn.net/weixin_67859959/article/details/135209645 函数和存储过程非常类似&#xff0c;也是可以存储在 Oracle 数据库中的 PL/SQL代码块&#xff0c;但是有返回值。 可以把经常使用的功能定义为一个函数&#xff0c;就像系统…

数仓建模—逻辑数据模型

数仓建模—逻辑数据模型 数据模型是数据元素及其基于现实世界对象之间的关系的可视化表示。数据模型揭示并定义数据在业务流程中的连接方式,并支持创建高效的信息系统或应用程序。例如,在商业智能中,数据模型定义用户可以在其分析中使用哪种数据。 逻辑数据模型 (LDM Logi…

【C++ STL序列容器】array 数组

文章目录 【 1. 基本原理 】【 2. array 的创建 】2.1 不赋初值2.2 赋默认值2.3 赋指定值 【 3. array 的成员函数 】实例 【 1. 基本原理 】 array 是在 C 普通数组的基础上添加了一些成员函数和全局函数。在使用上&#xff0c;它 比普通数组更 安全&#xff0c;且效率并没…

以太网帧格式解析

以太网的正式标准是IEEE802.3&#xff0c;它规定了以太网传输的帧结构。 以太网帧格式如下图所示&#xff1a; 以太网传输数据时&#xff0c;是按照上图的格式&#xff0c;自左到右依次传输的。需要注意的是前导码和SFD不属于以太网协议的内容&#xff0c;应该是属于物理层数据…

Java集合-List

目录 1、List 1.1 List集合特有方法 1.1.1 增加 1.1.2 删除 1.1.3 修改 1.1.4 查找 1.2 ArrayList 1.3 LinkedList 1.4 Vector 1、List ---| Iterable 接口 Iterator iterator() ----| Collection 接口 ------| List 接口 元素可以重复…

学习ArkTS -- 状态管理

装饰器 State 在声明式UI中&#xff0c;是以状态驱动试图更新&#xff1a; 状态&#xff08;State&#xff09;&#xff1a;指驱动视图更新的数据&#xff08;被装饰器标记的变量&#xff09; 视图&#xff08;View&#xff09;&#xff1a;基于UI描述渲染得到用户界面 说明…

病理验证mIF和TMA路线(自学)

目录 技术 使用配对病理切片 mIF验证 单基因使用TMA验证 技术 多重荧光免疫组化技术 (Multiplex immunohistochemical&#xff0c;mIHC) 也称作酪氨酸信号放大 (Tyramide dignal amplification&#xff0c;TSA) 技术&#xff0c;是一类利用辣根过氧化酶 (Horseradish Pero…

MyBatis 面试题(六)

1. MyBatis 有几种分页方式&#xff1f; MyBatis 的分页方式主要可以分为两大类&#xff1a;逻辑分页和物理分页。 逻辑分页是一次性把全部数据查询加载进内存&#xff0c;然后再进行分页。这种方式减少了IO次数&#xff0c;适合频繁访问、数据量少的情况&#xff0c;但不适合…

weblogic反序列化漏洞(CVE-2017-10271)复现

直接用vuluhub搭建现成的靶场做 访问靶场 打开是这样表示成功 想反弹shell 就先开启kali1的nc监听&#xff0c;这就监听2233端口吧 linux&#xff1a;nc -l -p 2233 抓包修改为攻击数据包 ip和端口可以任意修改 反弹的shell 还可以写入文件shell 只需要把提供的poc POS…

【代码随想录】day44

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 一、完全背包理论基础二、518零钱兑换 II三、377组合总和 Ⅳ总结 一、完全背包理论基础 完全背包&#xff1a;每个物品可以使用无数次。 二、518零钱兑换 II cla…

修复Windows搜索不工作的几种方法,总有一种适合你

序言 Windows搜索是Windows 10中一个非常有用的功能,它允许你搜索特定的程序、应用程序、文档、图片、文件、设置等,以便快速访问它们。但有时Windows搜索找不到我们预期的结果,甚至没有响应,这会给Windows用户带来很多不便。如果Windows 10中的搜索栏不工作,该怎么办?你…

【MySQL】SQL优化

SQL优化 插入数据 insert 一次插入数据和批量插入数据 insert into tb_test (id, name) values (1,Tom); insert into tb_test (id, name) values (1,Tom),(2,Jack),(3,Jerry);优化方案&#xff1a; 手动控制事务&#xff0c;且按主键顺序插入。start transaction; insert …

机器人实验室LAAS-CNRS介绍

一、LAAS-CNRS介绍 1、缩写介绍 同样的&#xff0c;给出英文缩写的全称&#xff0c;以便理解。这里的LAAS&#xff08;Laboratory for Analysis and Architecture of Systems&#xff09;指法国的系统分析与架构实验室&#xff0c;CNRS&#xff08;Centre National de la Rec…

[leetcode]jump-game-iv

. - 力扣&#xff08;LeetCode&#xff09; 给你一个整数数组 arr &#xff0c;你一开始在数组的第一个元素处&#xff08;下标为 0&#xff09;。 每一步&#xff0c;你可以从下标 i 跳到下标 i 1 、i - 1 或者 j &#xff1a; i 1 需满足&#xff1a;i 1 < arr.lengt…