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

2024-04-23 03:52

本文主要是介绍深入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;
}

这篇关于深入C语言,发现多样的数据之枚举和联合体的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/927712

相关文章

大模型研发全揭秘:客服工单数据标注的完整攻略

在人工智能(AI)领域,数据标注是模型训练过程中至关重要的一步。无论你是新手还是有经验的从业者,掌握数据标注的技术细节和常见问题的解决方案都能为你的AI项目增添不少价值。在电信运营商的客服系统中,工单数据是客户问题和解决方案的重要记录。通过对这些工单数据进行有效标注,不仅能够帮助提升客服自动化系统的智能化水平,还能优化客户服务流程,提高客户满意度。本文将详细介绍如何在电信运营商客服工单的背景下进行

基于MySQL Binlog的Elasticsearch数据同步实践

一、为什么要做 随着马蜂窝的逐渐发展,我们的业务数据越来越多,单纯使用 MySQL 已经不能满足我们的数据查询需求,例如对于商品、订单等数据的多维度检索。 使用 Elasticsearch 存储业务数据可以很好的解决我们业务中的搜索需求。而数据进行异构存储后,随之而来的就是数据同步的问题。 二、现有方法及问题 对于数据同步,我们目前的解决方案是建立数据中间表。把需要检索的业务数据,统一放到一张M

关于数据埋点,你需要了解这些基本知识

产品汪每天都在和数据打交道,你知道数据来自哪里吗? 移动app端内的用户行为数据大多来自埋点,了解一些埋点知识,能和数据分析师、技术侃大山,参与到前期的数据采集,更重要是让最终的埋点数据能为我所用,否则可怜巴巴等上几个月是常有的事。   埋点类型 根据埋点方式,可以区分为: 手动埋点半自动埋点全自动埋点 秉承“任何事物都有两面性”的道理:自动程度高的,能解决通用统计,便于统一化管理,但个性化定

使用SecondaryNameNode恢复NameNode的数据

1)需求: NameNode进程挂了并且存储的数据也丢失了,如何恢复NameNode 此种方式恢复的数据可能存在小部分数据的丢失。 2)故障模拟 (1)kill -9 NameNode进程 [lytfly@hadoop102 current]$ kill -9 19886 (2)删除NameNode存储的数据(/opt/module/hadoop-3.1.4/data/tmp/dfs/na

异构存储(冷热数据分离)

异构存储主要解决不同的数据,存储在不同类型的硬盘中,达到最佳性能的问题。 异构存储Shell操作 (1)查看当前有哪些存储策略可以用 [lytfly@hadoop102 hadoop-3.1.4]$ hdfs storagepolicies -listPolicies (2)为指定路径(数据存储目录)设置指定的存储策略 hdfs storagepolicies -setStoragePo

Hadoop集群数据均衡之磁盘间数据均衡

生产环境,由于硬盘空间不足,往往需要增加一块硬盘。刚加载的硬盘没有数据时,可以执行磁盘数据均衡命令。(Hadoop3.x新特性) plan后面带的节点的名字必须是已经存在的,并且是需要均衡的节点。 如果节点不存在,会报如下错误: 如果节点只有一个硬盘的话,不会创建均衡计划: (1)生成均衡计划 hdfs diskbalancer -plan hadoop102 (2)执行均衡计划 hd

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

深入探索协同过滤:从原理到推荐模块案例

文章目录 前言一、协同过滤1. 基于用户的协同过滤(UserCF)2. 基于物品的协同过滤(ItemCF)3. 相似度计算方法 二、相似度计算方法1. 欧氏距离2. 皮尔逊相关系数3. 杰卡德相似系数4. 余弦相似度 三、推荐模块案例1.基于文章的协同过滤推荐功能2.基于用户的协同过滤推荐功能 前言     在信息过载的时代,推荐系统成为连接用户与内容的桥梁。本文聚焦于

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl