C语言-联合和枚举

 

------------------------------------

---------------

------

最慢的步伐不是跬步,而是徘徊;

最快的脚步不是冲刺,而是坚持。 

今天来到我们的联合和枚举类型的讲解:

目录

联合体类型

联合体类型的声明

联合体类型的特点 

联合体大小的计算 

联合体类型的应用 

枚举类型

枚举类型的声明

枚举类型的特点

枚举类型的运用 


联合体类型

联合体类型的声明

1.像结构体⼀样,联合体也是由⼀个或者多个成员构成,这些成员可以不同的类型
2.但是编译器只为最大的成员分配足够的内存空间。联合体的特点是所有成员共用同⼀块内存空间。所以联合体也叫:共用体。

以下就是联合体的类型声明: 

union Un
{
 char a;
 int b;
 double c;
};

联合体类型的特点 

让我们看看系统给联合体分配的内存空间有多大:

#include<stdio.h>
union Un
{
	char a;
	int b;
	double c;
};
int main()
{
	union Un u = { 0 };
	printf("%d", sizeof(u));
	return 0;
}

 

我们可以把联合体的成员地址都打出来看看,是不是占用同一块空间:

#include <stdio.h>
union Un
{
	char a;
	int b;
	double c;
};
int main()
{
	union Un u = { 0 };
	printf("%p\n", &(u.a));
	printf("%p\n", &(u));
	printf("%p\n", &(u.c));
	printf("%p\n", &u);
	return 0;
}

 

我们发现4个地址一模一样 

 我们可以也用代码测试一下a在b中存储的位置: 

#include <stdio.h>
union Un
{
	char a;
	int b;
};
//联合体共用内存为4个字节
int main()
{
	union Un u = { 0 };
	u.b = 0x11223344;//将十六进制的数赋值给成员b
	u.a = 0x55;      //将十六进制的数赋值给成员a
	printf("%x\n", u.b);//打印b看看a在b中存储的位置
	return 0;
}

我们发现b的第4个字节被修改为55了 

 

总结:

1.我们可以看到联合体并不会像结构体那样给每个成员分配内存空间

2.联合的成员是共用同⼀块内存空间的,这样⼀个联合变量的大小,至少是最大成员的大小(因为联合至少得有能力保存最大的那个成员)

3.因为减少了内存上的占用,程序运行效率提高了,这就是联合体带来的优点

 以下是联合体和结构体的内存比较:

#include <stdio.h>
struct St
{
	char c;
	int i;
};
union Un
{
	char c;
	int i;
};
int main()
{
	struct St s = { 0 };
	union Un u = { 0 };
	printf("%d\n", sizeof(s));
	printf("%d\n", sizeof(u));
	return 0;
}

因为结构体存在内存对齐原则所以会浪费空间

那么问题来了,联合体的大小就一定是最大成员的大小吗?

其实并不是,联合体也存在内存对齐 

联合体大小的计算 

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

是的,联合体也存在内存对齐 ,我们来看以下代码:

#include <stdio.h>
union Un1
{
	char c[5];//成员大小为5,对齐数为1
	int i;    //成员大小为4,对齐数为4,最大对齐数为4
};
union Un2
{
	short c[7];//成员大小为14,对齐数为2
	int i;     //成员大小为4,对齐数为4,最大对齐数是4
};
int main()
{
	printf("%d\n", sizeof(union Un1));
	printf("%d\n", sizeof(union Un2));
	return 0;
}

Un1、Un2的大小是多少呢?

虽然联合体存在对齐数会浪费一定的空间

但相比较结构体这些浪费的空间不算什么 

结论:使用联合体是可以节省空间

联合体类型的应用 

使用联合体是可以节省空间的,举例:

比如,我们要搞⼀个活动,要上线⼀个礼品兑换单,礼品兑换单中有三种商品:图书、杯子、衬衫。

每⼀种商品都有:库存量、价格、商品类型和商品类型相关的其他信息。

图书:书名、作者、页数

杯⼦:设计

衬衫:设计、可选颜色、可选尺寸

那我们不耐心思考,直接写出一下结构:

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

	//特殊属性
	char title[20];//书名
	char author[20];//作者
	int num_pages;//⻚数

	char design[30];//设计
	int colors;//颜⾊
	int sizes;//尺⼨
};

上述的结构其实设计的很简单,起来也方便,但是结构的设计中包含了所有礼品的各种属性,这样使得结构体的大小就会偏大,比较浪费内存。因为对于礼品兑换单中的商品来说,只有部分属性信息是常用的。

比如: 商品是图书,就不需要design、colors、sizes。

所以我们就可以把公共属性单独写出来剩余属于各种商品本身的属性使用联合体起来,这样就可以介绍所需的内存空间,⼀定程度上节省了内存。

struct gift_list
{
 int stock_number;//库存量
 double price; //定价
 int item_type;//商品类型
 
 union{
 struct
 {
 char title[20];//书名
 char author[20];//作者
 int num_pages;//⻚数
 }book;
 struct
 {
 char design[30];//设计
 }mug;
 struct
 {
 char design[30];//设计
 int colors;//颜⾊
 int sizes;//尺⼨
 }shirt;
 }item;
};

枚举类型

枚举类型的声明

顾名思义,枚举的作用就是一一列举 ,以下是用枚举类型声明星期:

enum Sex//性别
{
	MALE,
	FEMALE,
	SECRET
};
#include<stdio.h>
enum Sex//性别
{
	MALE,
	FEMALE,
	SECRET
};
int main()
{
	printf("%d\n",MALE);
	printf("%d\n", FEMALE);
	printf("%d\n", SECRET);
	return 0;
}

 我们来看看MALE,FEMALE,SECRET的值分别是多少

1.{ }中的内容是枚举类型的 可能取值 ,是常量也叫 枚举常量, 是不能修改的 
2.这些可能取值都是有值的,默认从0开始,依次递增1
3.当然在声明枚举类型的时候 也可以赋初值
#include<stdio.h>
enum Sex//性别
{
	MALE = 2,
	FEMALE = 4,
	SECRET = 6
};

 枚举类型的特点

我们可以使用 #define 定义常量,为什么非要使用枚举?
枚举的优点:
1. 增加代码的可读性可维护性
2. 和#define定义的标识符比较枚举有类型检查,更加严谨
3. 便于调试,预处理阶段会删除 #define 定义的符号
4. 使用方便,一次可以定义多个常量
5. 枚举常量是遵循作⽤域规则的,枚举声明在函数内,只能在函数内使用

枚举类型的运用 

1.枚举变量只能由设置的枚举常量进行赋值:

#include<stdio.h>
enum Sex//性别
{
	MALE = 2,
	FEMALE = 4,
	SECRET = 6
};
int main()
{
	enum Sex s = MALE;
	return 0;
}

 2.可以对一些操作寓意表现得非常直观,增加代码的可读性:

 
void menu()
{
	printf("********************************************************************************************\n");
	printf("****************************              扫雷游戏             *****************************\n");
    printf("********************************************************************************************\n");
    printf("****************************                 1.启动            *****************************\n");
    printf("********************************************************************************************\n");
    printf("****************************                 0.结束            *****************************\n");
    printf("********************************************************************************************\n");
    printf("********************************************************************************************\n");
}
int main()
{
    int input = 0;                       //定义一个变量来来接收菜单
    srand((unsigned int)time(NULL));     //设置随机数
    do
    {
        menu();                           //添加扫雷菜单
        printf("请选择游戏的状态>:\n");
        scanf_s("%d",&input);
        switch (input){
            case 1:
                game();
                break;
            case 0:
                printf("退出游戏!\n");
                break;
        default:
            printf("输入错误,重新输入!\n");
            break;
        }
      } while (input);
    return 0;
}

我把我之前写过的扫雷代码放出来

如果不看菜单menu的话,你是不是一脸懵

1代表什么,0代表什么

接下来我将会用枚举类型对上面这段代码进行优化:

void menu()
{
    printf("********************************************************************************************\n");
    printf("****************************              扫雷游戏             *****************************\n");
    printf("********************************************************************************************\n");
    printf("****************************                 1.EXIT              *****************************\n");
    printf("********************************************************************************************\n");
    printf("****************************                 0.PLAY              *****************************\n");
    printf("********************************************************************************************\n");
    printf("********************************************************************************************\n");
}
enum Option
{
    EXIT,
    PLAY
};

int main()
{
    int input = 0;                       //定义一个变量来来接收菜单
    srand((unsigned int)time(NULL));     //设置随机数
    do
    {
        menu();                           //添加扫雷菜单
        printf("请选择游戏的状态>:\n");
        scanf_s("%d", &input);
        switch (input) {
        case EXIT:
            game();
            break;
        case PLAY:
            printf("退出游戏!\n");
            break;
        default:
            printf("输入错误,重新输入!\n");
            break;
        }
    } while (input);
    return 0;
}

通过本篇博客

我相信大家对C语言联合枚举都有了一定的理解

希望这些知识可以运用在你的实际生活中 

 

 

 

 

 

 

 

 

 

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

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

相关文章

Wireshark抓包分析RTMP协议时,出现Unknown问题

进行rtmp推流时&#xff0c;使用wireshark抓包&#xff0c;发现部分包显示Unknown 解决方法&#xff1a; 编辑 -> 首选项 -> Protocols -> RTMPT&#xff0c;这里Maximum packet size默认是32768 将该值调大&#xff0c;比如调成1048576&#xff0c;即可解决该问题。…

ChatGPT 的 18 种玩法,你还不会用吗?

你确定&#xff0c;你会使用 ChatGPT 了吗&#xff1f; 今天给大家整理了 18 种 ChatGPT 的用法&#xff0c;看看有哪些方法是你能得上的。 用之前我们可以打开R5Ai平台&#xff0c;可以免费使用目前所有的大模型 地址&#xff1a;R5Ai.com 语法更正 用途&#xff1a;文章…

改进LiteOS中物理内存分配算法(详细实验步骤+相关源码解读)

一、实验要求 优化TLSF算法&#xff0c;将Best-fit策略优化为Good-fit策略&#xff0c;进一步降低时间复杂度至O(1)。 优化思路&#xff1a; 1.初始化时预先为每个索引中的内存块挂上若干空闲块&#xff0c;在实际分配时避免分割&#xff08;split&#xff09;操作&#xff…

[原创]C++98升级到C++20的复习旅途-从汇编及逆向角度去分析“constexpr“关键字

[简介] 常用网名: 猪头三 出生日期: 1981.XX.XXQQ: 643439947 个人网站: 80x86汇编小站 https://www.x86asm.org 编程生涯: 2001年~至今[共22年] 职业生涯: 20年 开发语言: C/C、80x86ASM、PHP、Perl、Objective-C、Object Pascal、C#、Python 开发工具: Visual Studio、Delphi…

AtCoder Beginner Contest 331 题解 A-E

目录 A - TomorrowB - Buy One Carton of MilkC - Sum of Numbers Greater Than MeD - Tile PatternE - Set Meal A - Tomorrow 原题链接 题目描述 已知一年有M个月D天&#xff0c;求出第y年m月d天的后一天是哪一天。 思路&#xff1a;分类讨论 分别讨论m和d的是否是最后一个月…

基于SpringBoot的旅游信息网【源码好优多】

简介 旅游信息网是一款介绍旅游相关内容的网站&#xff0c;分为前台和后台部分&#xff0c;其中前台用户注册以后可以浏览景点、景点详情、预订景点、酒店、车票、保险、以及浏览旅游攻略、个人信息修改、在线留言等&#xff0c;管理员在后台对景点、攻略、订单信息、酒店信息、…

oj赛氪练习题

数组调整 import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner scanner new Scanner(System.in);int n scanner.nextInt();int k scanner.nextInt();int[] arr new int[n];for (int i 0; i < n; i) {arr[i] scanner.nextIn…

java源码-类与对象

1、面向对象与面向过程 在了解类和对象之前我们先了解一下什么是面向过程和面向对象。 1&#xff09;面向过程编程&#xff1a; C语言就是面向过程编程的&#xff0c;关注的是过程&#xff0c;分析出求解问题的步骤&#xff0c;通过函数调用逐步解决问题。 2&#xff09;面向对…

Redis 发布订阅机制深入探索

Redis 的发布订阅&#xff08;pub/sub&#xff09;机制是一种消息传递模式&#xff0c;允许消息的发送者&#xff08;发布者&#xff09;和消息的接收者&#xff08;订阅者&#xff09;通过一个中介层&#xff08;频道&#xff09;进行通信&#xff0c;而无需彼此直接交互。以下…

半导体工艺发展概述

集成电路发展到今天&#xff0c;经历从1940年的PN结发现&#xff0c;到1950年BJT三极管发明&#xff0c;再到1963年CMOS电路发明。从单纯基于Si的半导体电路&#xff0c;再到GaAs, GaN&#xff0c;SiGe, InP等化合物半导体集成电路。不断的通过化学材料配比&#xff0c;基本单元…

TinyVue 组件库助力赛意信息获得工业软件种子奖

首先恭喜广州赛意信息科技股份有限公司荣获工业软件种子奖&#xff01;在本次大赛中&#xff0c;凭借“数据驱动智造&#xff0c;基于 iDME 的赛意新一代 SMOM 赋能电子行业制造运营管理解决方案”这一作品脱颖而出~ 大赛简介 10月30日至10月31日&#xff0c;由广东省工业和信…

圆通速递查询入口,以表格的形式导出单号的每一条物流信息

批量查询圆通速递单号的物流信息&#xff0c;以表格的形式导出单号的每一条物流信息。 所需工具&#xff1a; 一个【快递批量查询高手】软件 圆通速递单号若干 操作步骤&#xff1a; 步骤1&#xff1a;运行【快递批量查询高手】软件&#xff0c;并登录 步骤2&#xff1a;点击…

Hadoop——分布式计算MapReduce和资源调度Yarn

分布式计算 MapReduce YARN架构 YARN集群部署 一、Hadoop安装目录下/etc/hadoop修改mapred-env配置文件&#xff0c;mapred-site.xml文件 二、etc/hadoop文件内&#xff0c;修改yarn-env.sh&#xff0c;yarn-site.xml 三、将配置好的文件分发到其他服务节点 start-dfs.…

SLAM ORB-SLAM2(10)轨迹跟踪过程

SLAM ORB-SLAM2(10)轨迹跟踪过程 1. 总体过程2. ORB 特征点提取2.1. 相机数据处理2.1.1. 单目相机图像处理2.1.2. 双目相机图像处理2.1.3. RGBD相机图像处理2.2. ORB 特征点3. 地图初始化3.1. 坐标形式3.2. 坐标原点3.3. 地图尺度4. 相机位姿初始估计4.1. 关键帧4.2. 运动模型…

文件搜索神器—Everything,结合内网穿透秒变在线搜索神器!

Everythingcpolar搭建在线资料库&#xff0c;实现随时随地访问 文章目录 Everythingcpolar搭建在线资料库&#xff0c;实现随时随地访问前言1.软件安装完成后&#xff0c;打开Everything2.登录cpolar官网 设置空白数据隧道3.将空白数据隧道与本地Everything软件结合起来总结 前…

【每日一题】1423. 可获得的最大点数-2023.12.3

题目&#xff1a; 1423. 可获得的最大点数 几张卡牌 排成一行&#xff0c;每张卡牌都有一个对应的点数。点数由整数数组 cardPoints 给出。 每次行动&#xff0c;你可以从行的开头或者末尾拿一张卡牌&#xff0c;最终你必须正好拿 k 张卡牌。 你的点数就是你拿到手中的所有…

【Android】解决安卓中并不存在ActivityMainBinding

安卓中并不存在ActivityMainBinding这个类&#xff0c;这个类是在XML布局的最外层加入就会自动生成。但是你在最后绑定主布局时会报错获取不到根节点getRoot(). 最好的办法就是&#xff0c;删除原来的最外层节点&#xff0c;再重新添加&#xff0c;感觉是因为复制时并没有让系…

[蓝桥杯 2020 省 AB1] 解码

做题前思路&#xff1a; 1.因为是多组输入&#xff0c;又包含字符于是我们可以先定义一个char类型数组arr 2.定义数组的长度&#xff1a;题目说简写&#xff08;字母加数字&#xff09;长度不超过100&#xff0c;但原来的长度可能超过100&#xff0c;加上小明不会将连续超过9…

Pandas时序数据分析实践—基础(1)

目录 1. Pandas基本结构2. Pandas数据类型2.1. 类型概述2.1.1. 整数类型&#xff08;int&#xff09;&#xff1a;2.1.2. 浮点数类型&#xff08;float&#xff09;&#xff1a;2.1.3. 布尔类型&#xff08;bool&#xff09;&#xff1a;2.1.4. 字符串类型&#xff08;object&a…

树_对称二叉树

//给你一个二叉树的根节点 root &#xff0c; 检查它是否轴对称。 // // // // 示例 1&#xff1a; // // //输入&#xff1a;root [1,2,2,3,4,4,3] //输出&#xff1a;true // // // 示例 2&#xff1a; // // //输入&#xff1a;root [1,2,2,null,3,null,3] //输出…