C语言(联合和枚举)

Hi~!这里是奋斗的小羊,很荣幸您能阅读我的文章,诚请评论指点,欢迎欢迎 ~~
💥个人主页:奋斗的小羊
💥所属专栏:C语言

在这里插入图片描述

🚀本系列文章为个人学习笔记,在这里撰写成文一为巩固知识,二为展示我的学习过程及理解。文笔、排版拙劣,望见谅。


目录

  • 前言
  • 一、联合体、
      • 1、联合体类型的声明
      • 2、联合体的特点
      • 3、相同成员的结构体和联合体对比
      • 4、联合体大小的计算
      • 5、联合练习
  • 二、枚举类型
      • 1、枚举类型的声明
      • 2、枚举类型的优点
      • 3、枚举类型的使用
  • 总结

前言

自定义类型除了结构体,还有联合体(共用体)、枚举等,本篇文章将对联合体、枚举展开详细介绍,讨论其特点,以及相较于结构体而言联合体又有什么相同之处和相异之处


一、联合体、

1、联合体类型的声明

联合体类型的关键字是:union

联合体和结构体是非常相似的,联合体也是由一个或多个成员组成,这些成员也可以是不同的类型。

//结构体
struct S
{
	int n;
	char c;
};
//联合体
union U
{
	int n;
	char c;
};

但是与结构体不同的是编译器只为联合体最大的成员分配足够的内存空间,联合体所有成员共用一块内存,因此联合体还有一个名字——共用体
正是因为联合体所有成员公共一块内存,所以当联合体其中一个成员的值变化时其他成员的值也跟着变化。


2、联合体的特点

#include <stdio.h>

struct S
{
	int n;//4  8  4
	char c;//1  8  1
	//8个字节
};

union U
{
	int n;
	char c;
};

int main()
{
	printf("%zd\n", sizeof(struct S));
	printf("%zd\n", sizeof(union U));
	return 0;
}

在这里插入图片描述

当我们计算相同成员的结构体和联合体的大小时,发现联合体确实只为最大成员开辟了足够的空间。
联合体的成员是共用同一块内存空间的,这样一个联合变量的大小,至少是最大成员的大小,因为联合至少得有能力保存最大的那个成员

union U
{
	int n;
	char c;
};

int main()
{
	union U u = { 0 };
	printf("%zd\n", sizeof(union U));
	printf("%p\n", &(u));
	printf("%p\n", &(u.n));
	printf("%p\n", &(u.c));
	return 0;
}

在这里插入图片描述

  • 联合体类型变量的创建和成员的引用类似于结构体

取出联合变量u的地址和两个成员的地址,可以看到两个成员确实是共用同一块内存空间的。
所以联合体叫共用体更为直观一点。
既然共用体成员一个变化其他的也跟着变化,这种特点有什么用呢?

  • 这使得共用体成员在同一时间只能使用一个,并不能多个同时出现。

3、相同成员的结构体和联合体对比

来看相同成员的结构体和联合体内存分布情况

//结构体
struct S
{
	int n;
	char c;
};
//联合体
union U
{
	int n;
	char c;
};

在这里插入图片描述
在VS上我们也可以证明这件事
在这里插入图片描述


4、联合体大小的计算

联合体的大小是不是就是最大成员的大小呢?

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

所以说联合体也是存在内存对齐的,那它的大小就不能单纯的认为是最大成员的大小了。
练习1

#include <stdio.h>

union U
{
	char arr[5];//1  8  1
	int n;      //4  8  4
	//最大成员5个字节
};

int main()
{
	printf("%zd\n", sizeof(union U));
	return 0;
}

在这里插入图片描述
练习2

#include <stdio.h>

union U
{
	short arr[7];//2  8  2
	int n;//4  8  4
	//最大成员14个字节
};

int main()
{
	printf("%zd\n", sizeof(union U));
	return 0;
}

在这里插入图片描述
通过上面的内容可以知道,联合体也是存在空间浪费的,但更多的是节省空间
举一个生活中可能出现的实际例子:
假如我们现在要搞一个礼品兑换单,其中有三种商品,图书、杯子、衬衫。
每一种商品都有:库存量、价格、商品类型和商品类型相关的其他信息。

  • 图书:书名、作者、页数
  • 杯子:颜色
  • 衬衫:设计、颜色、尺寸

如果我们直接写一个结构体:

struct gift_list
{
	//公共属性
	int stock_number;//库存量
	double price;//价格
	int item_type;//商品类型

	//特殊属性
	char book_name[20];//书名
	char auther[20];//作者
	int pages;//页数

	char design[20];//设计
	int red;//颜色
	int sizes;//尺寸
};

上面的结构体是完全可以解决问题的。
但是,当我们用它来描述图书时,其成员设计、颜色、尺寸是用不上的,但是这几个成员确实占着内存;当我们用它来描述杯子时,书名、作者、页数、设计、尺寸也用不上,但是这几个成员也占着内存。
公共属性是一直都用的,但特殊属性并不需要一直都在。

但是当我们运用联合体分装图书、杯子、衬衫的特殊属性后,因为联合体成员共用同一块内存空间,不能同一时刻出现,所以描述图书时和图书没有关系的信息并不占内存。

struct gift_list
{
	//公共属性
	int stock_number;//库存量
	double price;//价格
	int item_type;//商品类型

	union //因为这些类型只在gift_list中用一次所以并没有写名字
	{
		struct
		{
			char book_name[20];//书名
			char auther[20];//作者
			int pages;//页数
		}book;
		struct
		{
			int red;//颜色
		}cup;
		struct
		{
			char design[20];//设计
			int red;//颜色
			int sizes;//尺寸
		}shirt;
	}item;
};

这里在举一个例子
假设张三和李四一个想做早餐生意一个想做晚餐生意,他们分别租了一个铺子,这就类似结构体
但他们商量好租一个铺子,早上时张三在这个铺子里做早操生意,晚上时李四在这个铺子里做晚餐生意,两个人只需支付一间铺子的租金就能完成两个人的生意,互不影响,这就类似与联合体


5、联合练习

写一个程序,判断当前机器是大端还是小端

#include <stdio.h>

int main()
{
	int a = 1;
	if (*(char*)&a == 1)
	{
		printf("小端\n");
	}
	else
	{
		printf("大端\n");
	}
	return 0;
}

上面是我们之前的代码

#include <stdio.h>

union U
{
	char a;
	int b;
};

int main()
{
	union U u = { 0 };
	u.b = 1;
	if (u.a == 1)
	{
		printf("小端\n");
	}
	else
	{
		printf("大端\n");
	}
	return 0;
}

上面是我们用联合体的特点来实现的

这里联合体的特点就恰好解决了我们怎么在4个字节中拿出第一个字节的问题


二、枚举类型

1、枚举类型的声明

在现实生活中,有些值是有限的,是可以一一列举出来的,比如:星期只有星期一到星期日,三原色只有红、绿、蓝,性别只有男、女和保密等。
这些数据的表示就可以使用枚举。枚举就是一一列举的意思。

枚举类型的关键字是enum

enum Sex//性别
{
	//Sex的三种可能取值
	MALE,
	FEMALE,
	SECRET
};
enum Color//颜色
{
	//三原色的三种可能取值
	RED,
	GREEN,
	BLUE
}

上面定义的 enum Sex, enum Color 就是枚举类型。{ }中的内容是枚举类型的可能取值,也叫枚举常量
枚举类型的用法:

#include <stdio.h>
enum Sex//性别
{
	//Sex的三种可能取值
	MALE,
	FEMALE,
	SECRET
};
int main()
{
	enum Sex sex1 = MALE;
	enum Sex sex2 = FEMALE;
	return 0;
}

上面用声明的枚举类型创建了两个枚举类型变量,我们可以并且只可以给这两个变量赋Sex的三种可能取值。也就是说我们给枚举类型赋值时赋的是它的可能取值。

这些可能取值有没有什么特点呢?
在这里插入图片描述
也就是说,枚举常量的值默认是从0开始的,依次递增。
并且其值也就可以修改的。枚举常量的值会根据它前面的值而递增1。
在这里插入图片描述


2、枚举类型的优点

因为数字0、1、2并没有实际的意义,枚举类型可以让我们使用MALE的时候就表示数字0,使用FEMALE的时候就表示数字1,这样代码的可读性就增加了。
枚举的作用就是给一些常量起一个名字,使它具有实际意义。
虽然宏定义也可以定义常量,但枚举类型相较于宏定义有更多的优点。

  • 增加代码的可读性和可维护性
  • 和#define定义的标识符比较枚举有类型检查,更加严谨
  • 便于调试,预处理阶段会删除#define定义的符号
  • 使用方便,一次可以定义多个常量
  • 枚举常量是遵循作用域规则的,枚举声明在函数内,只能在函数内使用

3、枚举类型的使用

枚举类型的使用很简单,创建枚举类型,用枚举类型创建变量,再用枚举类型的可能取值给它赋值就行。
为了加深对枚举类型优点的理解,这里举一个我们之前写过的计算器的例子。

#include <stdio.h>
//只表示大概逻辑,并未完整
void menu()
{
	printf("################################\n");
	printf("########  1.add   2.sub  #######\n");
	printf("########  3.mul   4.div  #######\n");
	printf("########      0.exit     #######\n");
	printf("################################\n");
}

int main()
{
	int input = 0;
	do
	{
		menu();
		printf("请选择:");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			break;
		case 2:
			break;
		case 3:
			break;
		case 4:
			break;
		case 0:
			printf("退出计算器!\n");
			break;
		default:
			printf("选择错误,请重新选择!\n");
			break;
		}
	} while (input);
	return 0;
}

上面代码的逻辑是,选1表示算加法,选2表示算减法,选3表示算乘法,选4表示算除法。但是 case 1、case 2中的数字本身是没有实际意义的,当别人看这个代码时并不知道1234代表什么意思。
因此,在今天我们学了枚举后,可以定义枚举类型来解决这件事,提高代码的可读性。

#include <stdio.h>
//只表示大概逻辑,并未完整

enum option
{
	EXIT,//0
	ADD,//1
	SUB,//2
	MUL,//3
	DIV//4
};

void menu()
{
	printf("################################\n");
	printf("########  1.add   2.sub  #######\n");
	printf("########  3.mul   4.div  #######\n");
	printf("########      0.exit     #######\n");
	printf("################################\n");
}

int main()
{
	int input = 0;
	do
	{
		menu();
		printf("请选择:");
		scanf("%d", &input);
		switch (input)
		{
		case ADD:
			break;
		case SUB:
			break;
		case MUL:
			break;
		case DIV:
			break;
		case EXIT:
			printf("退出计算器!\n");
			break;
		default:
			printf("选择错误,请重新选择!\n");
			break;
		}
	} while (input);
	return 0;
}

我们用ADD,SUB,MUL,DIV代替1234就直观多了

总结

1.联合体就像一种特殊的结构体,它相较于结构体而言更节省空间,但也并不是完全节省,联合体也存在空间浪费。
2.枚举类型的出现在某些场景下很大提高了代码的可读性和可维护性,虽然在前期学习的过程中这种感觉可能并不明显,不过相信在以后积累起经验后会领略到枚举的魅力。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

00-macOS和Linux安装和管理多个Python版本

在 Mac 上安装多个 Python 版本可通过几种不同方法实现。 1 Homebrew 1.1 安装 Homebrew 若安装过&#xff0c;跳过该步。 /bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)" 1.2 安装 Python 如安装 Python …

Java多线程-初阶1

博主主页: 码农派大星. 数据结构专栏:Java数据结构 数据库专栏:MySQL数据库 JavaEE专栏:JavaEE 关注博主带你了解更多数据结构知识 1. 认识线程&#xff08;Thread&#xff09; 1.线程是什么 ⼀个线程就是⼀个 "执⾏流". 每个线程之间都可以按照顺序执⾏⾃⼰的代…

动态规划7:LCR 166. 珠宝的最高价值

动态规划解题步骤&#xff1a; 1.确定状态表示&#xff1a;dp[i]是什么 2.确定状态转移方程&#xff1a;dp[i]等于什么 3.初始化&#xff1a;确保状态转移方程不越界 4.确定填表顺序&#xff1a;根据状态转移方程即可确定填表顺序 5.确定返回值 题目链接&#xff1a;LCR …

sap 应用日志-Application Log

文章目录 sap 应用日志-Application Log概念事务代码函数创建程序显示配置文件运行结果弹出式全屏式 程序剖析清空日志创建日志模拟数据添加日志消息显示日志BAL_DSP_LOG_DISPLAY-显示内存消息 全部程序 sap 应用日志-Application Log 概念 SAP 应用日志&#xff08;Applicat…

别等了!速来体验 GLM-4-9B-Chat

昨日&#xff0c;智谱 AI 发布了基座大模型 GLM-4 的最新开源成果——GLM-4-9B&#xff0c;首次拥有了多模态能力。官方给出的数据显示&#xff0c;对比训练量更多的 Llama-3-8B 模型&#xff0c;GLM-4-9B 在中文学科方面的提升高达 50%&#xff0c;在多模态方面可以比肩 GPT-4…

新手上路:Linux虚拟机创建与Hadoop集群配置指南①(未完)

一、基础阶段 Linux操作系统: 创建虚拟机 1.创建虚拟机 打开VM,点击文件,新建虚拟机,点击自定义,下一步 下一步 这里可以选择安装程序光盘映像文件,我选择稍后安装 选择linux系统 位置不选C盘,创建一个新的文件夹VM来放置虚拟机,将虚拟机名字改为master方便后续识别…

白酒:茅台镇白酒的地域特色与环境优势

茅台镇&#xff0c;位于中国贵州省仁怀市&#xff0c;因其与众不同的自然环境和酿酒工艺而成为世界著名的白酒产区。作为茅台镇的品牌&#xff0c;云仓酒庄豪迈白酒以其卓着的品质和口感赢得了广大消费者的喜爱。而这一切&#xff0c;都离不开茅台镇的地域特色和环境优势。 茅台…

【星海出品】Jenkins

Jenkins部署环境 基本环境&#xff1a; 1.jdk环境&#xff0c;Jenkins是java语言开发的&#xff0c;因需要jdk环境。 2.git/svn客户端&#xff0c;因一般代码是放在git/svn服务器上的&#xff0c;我们需要拉取代码。 3.maven客户端&#xff0c;因一般java程序是由maven工程&a…

SQL Developer管理RESTful 服务

RESTful 服务依赖于ORDS&#xff08;Oracle REST Data Services&#xff09;&#xff0c;所以在进行本实验前&#xff0c;请先确认数据库服务器上的ORDS服务已启动&#xff1a; $ systemctl status ords ● ords.service - Oracle REST Data ServicesLoaded: loaded (/etc/sys…

YOLOv10(1):初探,训练自己的数据

目录 1. 写在前面 2. 值得关注的点 3. 训练自己的数据集 4. 阅读代码的小建议 1. 写在前面 很多人YOLOv9还没有完全研究透&#xff0c;YOLOv10出来了。 惊不惊喜&#xff0c;意不意外&#xff01; 据论文里提到&#xff0c;YOLOv10就是为了加速推理&#xff0c;在保证精度的…

天锐绿盾|防止源代码泄露系统,如何防止开发部门源码外泄?

#源代码防止泄露# 天锐绿盾是一款专为企业设计的数据安全软件&#xff0c;尤其擅长防止开发部门的源代码外泄&#xff0c;它通过一系列综合性的安全策略和技术手段实现这一目标。 PC地址&#xff1a; https://isite.baidu.com/site/wjz012xr/2eae091d-1b97-4276-90bc-6757c5d…

nesting in wrf

Choices for Nesting are:0 no nesting (only available for serial and smpar) 0. no nesting (only available for serial and smpar)1. basic2. preset moves preset moves3. vortex following • default is option 0 for serial/smpar, 1 for dmpar • smpar Shared Mem…

Linux环境---在线安装jdk

Linux环境—在线安装jdk 一、使用步骤 1.安装环境 JDK版本&#xff1a;1.8 1.1 建立存放软件的目录 注意&#xff1a;此处本人是将需要按照的软件存放在directory目录下&#xff0c;可根据实际情况调整接收路径。 命令如下&#xff1a; mkdir directory2.安装jdk 2.1 建…

Paperless-Ngx文档管理系统结合内网穿透实现随时远程搜索查阅文本

文章目录 前言1. 部署Paperless-ngx2. 本地访问Paperless-ngx3. Linux安装Cpolar4. 配置公网地址5. 远程访问6. 固定Cpolar公网地址7. 固定地址访问 前言 Paperless-ngx是一个开源的文档管理系统&#xff0c;可以将物理文档转换成可搜索的在线档案&#xff0c;从而减少纸张的使…

VS code上创建Vue项目详细化教程2-配置并创建Vue项目

目录 1. 环境准备 1.1 Node.js环境配置 1.1.1 安装Node.js 1.1.2 Node配置全局安装目录&#xff1a; 1.2 安装Vue-cli 2. 工程化Vue项目创建 2.1 命令行形式 2.2 UI 界面&#xff08;我们此处采用UI模式&#xff09; 2.2.1 在文件目录下终端输入 2.2.2 创建新项目 2…

Ansys Mechanical|为了提高结果精度而提高网格划分质量

一&#xff0e;高质量网格划分的要求 1. 一个好的网格划分可以捕获到所有重要几何细节。 2. 差的网格质量会导致收敛困难或者对物理场的描述不佳。 注&#xff1a;收敛困难有助于突出网格相关的误差。但是&#xff0c;如果结果不正确地收敛&#xff0c;则会在应用中导致各种…

Vuforia AR篇(六)— Mid Air 半空识别

目录 前言一、什么是Mid Air&#xff1f;二、使用步骤三、示例代码四、效果 前言 增强现实&#xff08;AR&#xff09;技术正在改变我们与数字世界的互动方式。Vuforia作为先进的AR开发平台&#xff0c;提供了多种工具来创造引人入胜的AR体验。其中&#xff0c;Mid Air功能以其…

纷享销客集成平台(IPaaS)解决方案

针对传统对接方式中的痛点&#xff0c;集成平台提炼了一套成熟的解决方案和配套工具。 痛点&#xff11;&#xff1a;编码工作量大。 每个功能点&#xff0c;和众多的容错分支&#xff0c;都需要逐行编码实现。日志打少了影响问题排查&#xff0c;打多了浪费大量存储。 集成…

Pinia的安装及使用

一、pinia是什么&#xff1f; Store (如 Pinia) 是一个保存状态和业务逻辑的实体&#xff0c;它并不与你的组件树绑定。换句话说&#xff0c;它承载着全局状态。它有点像一个永远存在的组件&#xff0c;每个组件都可以读取和写入它。它有三个概念&#xff0c;state、getter 和 …

【调试笔记-20240602-Linux-在 OpenWRT-23.05 上配置 frps 与 frpc 之间使用 TLS 进行传输】

调试笔记-系列文章目录 调试笔记-20240602-Linux-在 OpenWRT-23.05 上配置 frps 与 frpc 之间使用 TLS 进行传输 文章目录 调试笔记-系列文章目录调试笔记-20240602-Linux-在 OpenWRT-23.05 上配置 frps 与 frpc 之间使用 TLS 进行传输 前言一、调试环境操作系统&#xff1a;O…