【C语言】一篇文章深入解析联合体和枚举且和结构体的区别

在这里插入图片描述

文章目录

  • 📝前言
  • 🌠 联合体类型的声明
    • 🌉联合体的特点
  • 🌠相同成员的结构体和联合体对⽐
    • 🌉联合体⼤⼩的计算
  • 🌠联合体应用
    • 🌉枚举类型的声明
  • 🌠枚举类型的优点
    • 🌉 枚举类型的使⽤
  • 🚩总结


📝前言

联合体(union)是允许一个变量通过不同的接口访问内存的一种数据类型,表示一个变量可以存储不同类型的值,而枚举是使用enum关键字定义一组相关且互斥的整形常量集合。本章阿森将和你学习联合体类型的声明,特点,有关大小的计算,还有枚举类型的声明,优点和使用。文章干货满满!学习起来吧😃!

🌠 联合体类型的声明

同结构体一样,声明结构体类型需要使用struct关键字,联合体则用union关键字。

  1. 包含对象名的声明方式:
union 联合体名
{类型 成员1;类型 成员2;... 类型 成员n;
}对象名;
  • 代码理解:
#include <stdio.h>union S
{char c;int a;
}s1;int main()
{s1.c = 'a';printf("%c\n", s1.c);s1.a = 10;printf("%d\n", s1.a);return 0;
}

代码运行:
在这里插入图片描述

  1. 不包含对象名的声明格式:
union 类型名 
{ 类型 成员1;类型 成员2;... 类型 成员n;
};
  • 代码实现:
#include <stdio.h>union S
{char c;int a;
};int main()
{union S s2;s2.c = 'b';printf("%c\n", s2.c);s2.a = 20;printf("%d\n", s2.a);return 0;
}

运行:
在这里插入图片描述

🌉联合体的特点

  1. 编译器只为最⼤的成员分配⾜够的内存空间。联合体的特点是所有成员共⽤同⼀块内存空间。所以联合体也叫:共⽤体。

例如:

union u
{char c;int u;
};int main()
{union u uu;printf("联合体uu的大小为%zd\n", sizeof(uu));printf("   (uu)地址为%p\n", &uu);printf("&(uu.c)地址为%p\n", &(uu.c));printf("&(uu.u)地址为%p\n", &(uu.u));return 0;
}

输出:
在这里插入图片描述
图解:
在这里插入图片描述

  1. 联合的成员是共⽤同⼀块内存空间的,这样⼀个联合变量的⼤⼩,⾄少是最⼤成员的⼤⼩(因为联合⾄少得有能⼒保存最⼤的那个成员)
//联合类型的声明
union u
{char c;int i;
};
int main()
{//联合变量的定义union u uu = { 0 };un.i = 0x11223344;un.c = 0x55;printf("%x\n", uu.i);return 0;
}

输出:
在这里插入图片描述
图解:
在这里插入图片描述
union定义了intchar两个成员,共享同一块内存空间,int类型占4个字节,低地址在前,高地址在后,char类型只占1个字节,存储在int的低地址字节。当执行:uu.i = 0x11223344时,此时int的4个字节分别存储如图,然后执行: uu.c = 0x55,由于VS是小端存储,低字节放在低地址处char只占1个字节,它会覆盖int低地址的那个字节。所以int原来低地址字节0x44被覆盖为0x55

🌠相同成员的结构体和联合体对⽐

结构体和联合体的主要区别在于:

  • 结构体中每个成员占用自己独立的内存空间,可以同时访问每个成员。
  • 联合体中所有成员共享同一块内存空间,只能同时访问其中一个成员。
  1. 内存布局:
    结构体中每个成员都有固定的偏移地址,占用独立的内存空间。
    联合体中所有成员共享同一块内存,没有偏移地址,只能使用一个成员。

  2. 访问成员:
    结构体可以同时读取各个成员的值。
    联合体只能访问当前使用的成员,其他成员的值将被覆盖。

  3. 大小:
    结构体的大小是所有成员大小的和。
    联合体的大小至少是最大成员的大小。

  • 结构体:
struct S
{char c;int i;
};
struct S s = {0};
  • 联合体:
union u
{char c;int i;
};
union u uu = { 0 };

图解对比:
结构体S占用char + int+有可能开辟浪费的空间大小的内存,可以独立访问ci,联合体u只占用int大小的内存,访问ci时值会覆盖,结构体各成员独立,联合体成员共享同一内存空间。
在这里插入图片描述

🌉联合体⼤⼩的计算

点击可以查看结构体的内存对齐规则——>【C语言】自定义类型:结构体深入解析(二)结构体内存对齐&&宏offsetof计算偏移量&&结构体传参
联合体大小计算规则:

  1. 联合的⼤⼩⾄少是最⼤成员的⼤⼩。
  2. 当最⼤成员⼤⼩不是最⼤对⻬数的整数倍的时候,就要对⻬到最⼤对⻬数的整数倍。
  • 来代码理解:
union Un1
{char c[5];int i;
};union Un2
{short c[7];int i;
};int main()
{printf("%zd\n", sizeof(union Un1));//8printf("%zd\n", sizeof(union Un2));//16return 0;
}

运行:
在这里插入图片描述
图解分析:
在这里插入图片描述
首先看union Un1如果联合体的大小是最大成员的最大成员的的大小,在联合体union Un1中,char[5]的大小理应是5,那计算的结果不是5。为什么是8呢?这是因为它完成了对齐的操作,如果是数组,是按元素类型大小来算他的对齐数的。char 元素的类型大小是1VS默认对齐数是8,对齐数是8i 的大小是4VS默认对齐数是8,对齐数是4,接下来(4>1)整个联合体的对齐数是4,根据当最⼤成员⼤⼩不是最⼤对⻬数的整数倍的时候,就要对⻬到最⼤对⻬数的整数倍。此时最大成员大小是数组char [5]大小为5,5不是4的整数倍,84*2)是4的整数倍。是不是真的是这样呢?会不会是偶然呢?

接下来我们看第二组:union Un2首先short c[7]是数组,总大小为14,然后由于数组是按照元素的类型大小来算对齐数,类型为short类型大小为2VS默认对齐数为8,对齐数为22<8),i的大小是4VS默认对齐数是8,那么对齐数是44<8),然后整个联合体的对齐数是44>2),然后看成员最大对齐数的大小(short c[7]的大小是2*7=14)是不是整个联合体的对齐数(4)的整数倍,可见14不是4的整数倍, 根据第二条规则:当最⼤成员⼤⼩不是最⼤对⻬数的整数倍的时候,就要对⻬到最⼤对⻬数的整数倍。因此还要多用2个字节,升到164*4)个字节才是整数倍。
联合体的对齐规则与结构体相似:
点击可以查看结构体的内存对齐规则——>【C语言】自定义类型:结构体深入解析(二)结构体内存对齐&&宏offsetof计算偏移量&&结构体传参

🌠联合体应用

使⽤联合体是可以节省空间的:
⽐如,我们要搞⼀个活动,要上线⼀个礼品兑换单,礼品兑换单中有三种商品:图书、杯⼦、衬衫。
每⼀种商品都有:库存量、价格、商品类型和商品类型相关的其他信息。
图书:书名、作者、⻚数
杯⼦:设计
衬衫:设计、可选颜⾊、可选尺⼨

结构体表示:

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;//尺⼨
};

上述的结构其实设计的很简单,⽤起来也⽅便,但是结构的设计中包含了所有礼品的各种属性,这样使得结构体的⼤⼩就会偏⼤,⽐较浪费内存。因为对于礼品兑换单中的商品来说,只有部分属性信息是常⽤的。⽐如:商品是图书,就不需要designcolorssizes
所以我们就可以把公共属性单独写出来,剩余属于各种商品本⾝的属性使⽤联合体起来,这样就可以介绍所需的内存空间,⼀定程度上节省了内存。

联合体应用:

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;
};

练习:写⼀个程序,判断当前机器是⼤端?还是⼩端?

  1. 第一种方法:
int check_sys()
{int n = 1;//01 00 00 00     00 00 00 01return *(char*)&n;
}int main()
{int ret = check_sys();if (ret == 1)printf("小端\n");elseprintf("大端\n");return 0;
}

VS运行:
在这里插入图片描述

  1. 第二种联合体巧妙方法:
int check_sys()
{union{char c;int i;}u;u.i = 1;return u.c;//返回1是⼩端,返回0是⼤端
}

VS运行:

小端

图解:
在这里插入图片描述

大端存储:是指数据的低位字节内容保存在内存的高地址处,而数据的高位位字节内容,保存在内存的低地址处。
小端存储:是指数据的低位字节内容保存在内存的低地址处,而数据的高位字节内容,保存在内存的高地址处。

如果01是低位字节存储到低地址c时,是小端存储,如果01低位字节存储到高地址处,没有存储到c的位置,那么c的位置存储着00,返回为0,是大端存储。

🌉枚举类型的声明

枚举类型(enum)是一种特殊的类型,它可以为一组相关的常量值赋予用户定义的名称。
—>简单来说:枚举顾名思义就是⼀⼀列举。
枚举类型的声明语法:

enum 标识符 
{枚举常量1, 枚举常量2,...
} 变量;
  1. enum 关键字声明这是一个枚举类型。

  2. 标识符是枚举类型的名称。

  3. 在大括号{}内列出枚举类型的多个枚举常量,用逗号分隔。

  4. 变量是枚举类型的变量,可以直接使用枚举类型名或枚举常量初始化。

例如:

enum Color //Color是枚举类型名
{RED,     // 枚举常量  GREEN,BLUE
} color;///color是Color类型的变量int main()
{printf("%d\n", RED);printf("%d\n", GREEN);printf("%d\n", BLUE);return 0;
}

输出:
在这里插入图片描述

  • 枚举常量默认从0开始依次累加1。也可以手动为枚举常量赋值:

例如:

enum Color 
{RED = 1,GREEN = 2,BLUE = 4
}

运行结果:
在这里插入图片描述

  • 当然第一个元素未被赋值,给其它的常量赋值,该常量前面的值是默认值(0,1,2)后面递增1。

例如:

enum Color
{RED,white,GREEN = 8,BLUE ,BLACK,
};int main()
{printf("%d\n", RED);printf("%d\n", white);printf("%d\n", GREEN);printf("%d\n", BLUE);printf("%d\n", BLACK);return 0;
}

输出:
在这里插入图片描述

🌠枚举类型的优点

为什么使⽤枚举?
我们可以使⽤ #define 定义常量,为什么⾮要使⽤枚举?

枚举的优点:

  1. 增加代码的可读性和可维护性
    如:之前的扫雷中可以这样定义用PLAY代替1EXIT代替0,更具有个性化:
#include <stdio.h> 
#include <string.h>// 定义游戏选择枚举类型
enum Game_Selection
{EXIT, // 退出游戏PLAY  // 开始游戏
};// 打印菜单
void menu()
{printf("********** Menu **********\n");printf("PLAY - Start the game\n");printf("EXIT - Exit the game\n");printf("********** Menu **********\n");
}int main()
{enum Game_Selection input; // 声明游戏选择变量char choice[10]; // 声明选择输入缓冲区do{menu(); // 调用菜单函数printf("Please enter your choice: ");scanf("%s", choice); // 读取选择输入getchar(); // 清除输入缓冲区if (strcmp(choice, "PLAY") == 0){input = PLAY; // 设置选择为开始游戏}else if (strcmp(choice, "EXIT") == 0){input = EXIT; // 设置选择为退出游戏}else{printf("输入错误,请重新输入\n");continue; // 输入错误,继续循环}switch (input){case PLAY:printf("扫雷游戏启动!\n");break;case EXIT:printf("不玩了,启动不了!\n");break;}} while (input != EXIT);return 0;
}

代码运行:
在这里插入图片描述

  1. #define定义的标识符⽐较枚举有类型检查,更加严谨。

  2. 便于调试,预处理阶段会删除 #define 定义的符号在这里插入图片描述

  3. 使⽤⽅便,⼀次可以定义多个常量

  4. 枚举常量是遵循作⽤域规则的,枚举声明在函数内,只能在函数内使⽤

🌉 枚举类型的使⽤

那是否可以拿整数给枚举变量赋值呢?在C语⾔中是可以的,但是在C++是不⾏的,C++的类型检查⽐较严格。

  • C语言中,枚举类型实际上就是整数类型,编译器会把枚举常量替换成对应的整数值。所以可以用整数直接给枚举变量赋值。

  • 而在C++中,枚举类型是完全独立的类型。编译器会检查类型是否匹配,不允许用整数直接给枚举变量赋值。

例如:

# define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
//C语言
enum Color//颜⾊
{RED = 1,GREEN = 2,BLUE = 4
};
int main()
{enum Color c;c = 1; // 可以直接赋值整数return 0;
}

输出:
在这里插入图片描述

// C++语言
enum Color//颜⾊
{RED = 1,GREEN = 2,BLUE = 4
};Color c;
c = 1; // 错误,类型不匹配

输出:
在这里插入图片描述

总结:
C语言中枚举类型实际上就是整数,允许用整数直接赋值
C++中枚举类型是独立类型,不允许用整数直接赋值,需要强制类型转换


🚩总结

这次阿森和你一起学习联合体类型的声明,特点,然后进行相同成员的结构体和联合体对⽐,⼤⼩的计算,联合体应用,枚举类型的声明,优点和扫雷改造使⽤方法,阿森将下一节和你一起学习动态内存管理🚀 。

感谢你的收看,如果文章有错误,可以指出,我不胜感激,让我们一起学习交流,如果文章可以给你一个小小帮助,可以给博主点一个小小的赞😘
请添加图片描述

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

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

相关文章

方案:智能分析网关V4在高校实验室安全管理中的应用

一、方案背景 实验室作为科研与教学的核心场所&#xff0c;其重要性不言而喻。高校实验室由于其开放性与多样性&#xff0c;安全管理尤为重要。高校实验室的安全管理&#xff0c;不仅是保障科研与教学质量的基础&#xff0c;更是校园安全的重要组成部分。一旦发生安全事故&…

LabVIEW在电机噪声与振动探测的应用

LabVIEW在电机噪声与振动探测的应用 硬件部分是电机噪声和振动测试分析系统的基础&#xff0c;主要由三大核心组件构成&#xff1a;高灵敏度振动传感器、先进的信号调理电路和高性能数据采集卡。这些设备协同工作&#xff0c;确保了从电机捕获的噪声和振动信号的准确性和可靠性…

山西电力市场日前价格预测【2023-12-28】

日前价格预测 预测说明&#xff1a; 如上图所示&#xff0c;预测明日&#xff08;2023-12-28&#xff09;山西电力市场全天平均日前电价为814.30元/MWh。其中&#xff0c;最高日前电价为1500.00元/MWh&#xff0c;预计出现在08:00~08:45,17:00~20:15。最低日前电价为394.61元/…

Vue3-29-路由-编程式导航的基本使用

补充一个知识点 路由配置中的 name 属性 &#xff1a; 可以给你的 路由 指定 name属性&#xff0c;称之为 命名路由。 这个 name 属性 在 编程式导航 传参时有重要的作用。 命名路由的写法如下 &#xff1a; 像指定 path 一样&#xff0c;直接指定一个 name 属性即可。{path:/d…

ubuntu 安装apisix -亲测可用

官方未提供在ubuntu系统中安装apisix的方式&#xff0c;似乎只能通过源码方式安装&#xff0c;但是并不推荐&#xff0c;非常容易失败&#xff0c; 具体操作方式如下&#xff1a; ubuntu和Debian其实类似的&#xff0c;可使用DEB方式安装&#xff0c;如下截图 注意&#xff1…

实用的二进制文件分割器

自己写的一个能方便分割文件的小工具 1.可以按照任意方式分割文件 (1)分割范围 (2)分割块大小 (3)分割份数 (4)可以反向分割(从文件末尾向文件头分割) 2.可以指定输出文件名规则 (1)文件名前缀 (2)文件名序号 (3)文件名后缀(扩展名) (4)文件名…

Ansible Windows批量安装软件

文章目录 1&#xff1a;Windows配置WINRM2: ansible安装3&#xff1a;操作步骤3.1 配置主机清单3.2 测试ansible执行命令3.3 测试安装7Z ansible操作通过winrm协议windows&#xff0c;经过实践精简以下方法能快速配置&#xff0c;并能通过测试 更多文档参考: 支持的windows版本…

vscode 支持c,c++编译调试方法

概述&#xff1a;tasks.jason launch.json settings.json一定要有&#xff0c;没有就别想跑。还有就是c 和c配置有区别&#xff0c;切记&#xff0c;下文有说 1.安装扩展插件。 2.安装编译器&#xff0c;gcc.我用的是x86_64-8.1.0-release-win32-seh-rt_v6-rev0.7z &#xf…

Linux之缓冲区的理解

目录 一、问题引入 二、缓冲区 1、什么是缓冲区 2、刷新策略 3、缓冲区由谁提供 4、重看问题 三、缓冲区的简单实现 一、问题引入 我们先来看看下面的代码&#xff1a;我们使用了C语言接口和系统调用接口来进行文件操作。在代码的最后&#xff0c;我们还使用fork函数创建…

万界星空低代码云MES-才是工业MES的未来

万界星空科技作为一家在云MES系统的研发、生产自动化方面拥有很多年行业经验的科技型企业&#xff0c;多年来专注于云MES系统的研发与技术支持服务&#xff0c;目前已成为国内知名的智能制造整体解决方案提供商。 近几年&#xff0c;万界星空科技发掘制造行业生产及物流难点、…

图像的颜色及Halcon颜色空间转换transfrom_rgb/trans_to_rgb/create_color_trans lut

图像的颜色及Halcon颜色空间转换 文章目录 图像的颜色及Halcon颜色空间转换一. 图像的色彩空间1. RGB颜色 2. 灰度图像3. HSV/ HSI二. Bayer 图像三. 颜色空间的转换1. trans_from_rgb算子2. trans_to_rgb算子3. create_color_trans_lut算子 图像的颜色能真实地反映人眼所见的真…

挑战Python100题(8)

100+ Python challenging programming exercises 8 Question 71 Please write a program which accepts basic mathematic expression from console and print the evaluation result. 请编写一个从控制台接受基本数学表达式的程序,并打印评估结果。 Example: If the follo…

每日一练:LeeCode-20. 有效的括号(简)【栈】

给定一个只包括 ‘(’&#xff0c;‘)’&#xff0c;‘{’&#xff0c;‘}’&#xff0c;‘[’&#xff0c;‘]’ 的字符串 s &#xff0c;判断字符串是否有效。 有效字符串需满足&#xff1a; 左括号必须用相同类型的右括号闭合。 左括号必须以正确的顺序闭合。 每个右括号都有…

隧道代理HTTP工作原理:一场奇妙的网络魔法表演

嘿&#xff0c;小伙伴们&#xff01;今天我们要一起探索一个有趣的话题——隧道代理HTTP的工作原理。这不是普通的表演&#xff0c;而是一场奇妙的网络魔法表演&#xff01; 首先&#xff0c;让我们想象一下&#xff0c;网络世界就像一个大舞台&#xff0c;而我们每个人都是这…

Linux驱动开发简易流程

推荐视频&#xff1a; 正点原子【第四期】手把手教你学 Linux之驱动开发篇 小智-学长嵌入式Linux&Android底层开发入门教程 能力矩阵 基础能力矩阵 熟悉c/c、熟悉数据结构 熟悉linux系统&#xff0c;Shell脚本&#xff0c;Makefile/cmake/mk 文件IO、多线程、竞争、并发…

深度优先和广度优先

文章目录 前言一、深度和广度的区别二、代码演示1.准备数据,构造树2.深度优先遍历3.广度优先遍历 总结 前言 深度优先和广度优先的区别&#xff1a; 搜索方式不同 。深度优先搜索算法不全部保留结点&#xff0c;扩展完的结点从数据库中弹出删去&#xff1b;广度优先搜索算法需…

oracle-sga-shared_pool

shared pool 缓冲sql语句和执行计划 shared pool由三部分组成 free libray&#xff1a;缓存sql执行计划 row cathe &#xff1a;缓存数据字典 硬解析&#xff1a;1判断语法2判断对象是否存在3有没有权限4 从n个执行方案中选出最优解&#xff0c;生成执行计划&#xff0c;这一…

壮志酬筹>业务被裁>副业转正>收入回正。一个前黑马程序员老师的2023

从年初时的踌躇满志&#xff0c;到年中时整个业务线被砍。全职做前端训练营&#xff0c;四个多月的时间帮助100多名同学拿到了满意的offer&#xff0c;同时也让我的收入重归正轨。仅以这个视频记录我&#xff0c;一个普通程序员的 2023 。 视频版可直接访问 Hello&#xff0c;大…

基于ElementUI二次封装弹窗组件

效果&#xff1a; 一、自定义内容类型弹窗 <!-- title&#xff1a;对话框的标题confirmLoading&#xff1a;当前是否处于提交中titleCenter&#xff1a;对话框标题居中方式footerCenter&#xff1a;底部按钮的对其方式visible&#xff1a;是否显示弹窗width&#xff1a;设置…

JavaScript:正则表达式

JavaScript&#xff1a;正则表达式 什么是正则表达式正则表达式语法定义正则表达式判断是否有匹配的字符串查找匹配的字符串 正则表达式匹配法则元字符边界符量词字符类 什么是正则表达式 正则表达式用于匹配字符串中字符的组合模式。 正则表达式会依据其自身语法&#xff0c;…