当前位置: 首页 > news >正文

C语言中的联合和枚举(未完)

1、联合体

联合体类型的声明

        像结构体⼀样,联合体也是由⼀个或者多个成员构成,这些成员可以不同的类型。但是编译器只为最⼤的成员分配⾜够的内存空间。联合体的特点是所有成员共⽤同⼀块内存空间。所以联合体也叫:共⽤体。因为所有变量公用同一片区域,所以给联合体其中⼀个成员赋值,其他成员的值也跟着变化。
#include <stdio.h>
//联合类型的声明
union Un
{char c;int i;
};
int main()
{//联合变量的定义union Un un = {0};//计算连个变量的⼤⼩printf("%d\n", sizeof(un));return 0;
}输出结果: 4

联合体的特点

        联合体的成员是共用同一块内存空间的,这样一个联合变量的大小,至少是最大成员的大小(因为来拿赫梯至少要有能力保存最大的那个成员,至少的意思是可能有内存对齐的情况)。

//代码1
#include <stdio.h>
//联合类型的声明
union Un
{char c;int i;
};
int main()
{//联合变量的定义union Un un = {0};// 下⾯输出的结果是⼀样的吗?printf("%p\n", &(un.i));printf("%p\n", &(un.c));printf("%p\n", &un);return 0;
}输出:
001AF85C
001AF85C
001AF85C

        输出的三个地址一模一样,这也印证了之前说的联合体成员变量共用同一块地址。

//代码2
#include <stdio.h>
//联合类型的声明
union Un
{char c;int i;
};
int main()
{//联合变量的定义union Un un = {0};un.i = 0x11223344;un.c = 0x55;printf("%x\n", un.i);return 0;
}输出:
11223355

通过代码2的输出,我们发现将i的第四个字节内容修改为了55.

通过上面两段代码,我们可以活出un的内存布局:

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

//结构体struct S
{char c;int i;
};
struct S s = {0};//联合体union Un
{char c;int i;
};
union Un un = {0};

联合体大小的计算

联合的⼤⼩⾄少是最⼤成员的⼤⼩。
当最⼤成员⼤⼩不是最⼤对⻬数的整数倍的时候,就要对⻬到最⼤对⻬数的整数倍。
#include <stdio.h>
union Un1
{char c[5];int i;
};
union Un2
{short c[7];int i;
};
int main()
{//下⾯输出的结果是什么?printf("%d\n", sizeof(union Un1));printf("%d\n", sizeof(union Un2));return 0;
}

 输出结果:

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

联合的⼀个练习

之前我们讲过一个判断当前机器是大端小端的案例:

int check_sys()
{union{int i;char c;}un;un.i = 1;return un.c;//返回1是⼩端,返回0是⼤端
}

使用联合体同样可以进行判断,甚至比之前的代码还简单。

相关文章:

  • python爬虫基础实验:通过DBLP数据库获取数据挖掘顶会KDD在2023年的论文收录和相关作者信息
  • #微信小程序:微信小程序常见的配置传旨
  • 跨越时空的纽带:探索Facebook如何连接人与人
  • Lambda函数与Selenium WebDriverWait类一起使用
  • C++11 新特性:常量表达式 constexpr(下)
  • 【算法】雪花算法生成分布式 ID
  • C语言-结构体-015
  • 设计模式总结(四)
  • Spring 之声明式事务和 Spring Junit 案例应用详解
  • Spring Boot: 使用MongoOperations操作mongodb
  • Vue+SpringBoot打造智慧家政系统
  • 机器学习之无监督学习简介及算法库推荐
  • 学点Java_Day6_基于Copyright和Actions On Save的IDEA自动更新文件最后编辑时间的方法研究(IDEA2021.2及以上)
  • 联想笔记本的声音键没有反应怎么办?
  • Selenium 学习(0.21)——软件测试之单元测试
  • android 一些 utils
  • angular学习第一篇-----环境搭建
  • Cookie 在前端中的实践
  • CSS中外联样式表代表的含义
  • js ES6 求数组的交集,并集,还有差集
  • js数组之filter
  • Lsb图片隐写
  • Python 基础起步 (十) 什么叫函数?
  • Spring Cloud Feign的两种使用姿势
  • TypeScript迭代器
  • vue2.0项目引入element-ui
  • Vultr 教程目录
  • 阿里云容器服务区块链解决方案全新升级 支持Hyperledger Fabric v1.1
  • 可能是历史上最全的CC0版权可以免费商用的图片网站
  • 爬虫模拟登陆 SegmentFault
  • 让你成为前端,后端或全栈开发程序员的进阶指南,一门学到老的技术
  • 时间复杂度与空间复杂度分析
  • 线性表及其算法(java实现)
  • C# - 为值类型重定义相等性
  • Redis4.x新特性 -- 萌萌的MEMORY DOCTOR
  • Salesforce和SAP Netweaver里数据库表的元数据设计
  • ​Distil-Whisper:比Whisper快6倍,体积小50%的语音识别模型
  • ​一帧图像的Android之旅 :应用的首个绘制请求
  • (C语言)fread与fwrite详解
  • (Note)C++中的继承方式
  • (Repost) Getting Genode with TrustZone on the i.MX
  • (二开)Flink 修改源码拓展 SQL 语法
  • (附源码)spring boot网络空间安全实验教学示范中心网站 毕业设计 111454
  • (附源码)ssm高校社团管理系统 毕业设计 234162
  • (万字长文)Spring的核心知识尽揽其中
  • . Flume面试题
  • ./mysql.server: 没有那个文件或目录_Linux下安装MySQL出现“ls: /var/lib/mysql/*.pid: 没有那个文件或目录”...
  • .mkp勒索病毒解密方法|勒索病毒解决|勒索病毒恢复|数据库修复
  • .Net 6.0 处理跨域的方式
  • .NET 设计模式—简单工厂(Simple Factory Pattern)
  • .NET 使用 JustAssembly 比较两个不同版本程序集的 API 变化
  • .NET性能优化(文摘)
  • @RequestMapping 的作用是什么?
  • [ Linux ] git工具的基本使用(仓库的构建,提交)
  • [ 云计算 | AWS 实践 ] Java 如何重命名 Amazon S3 中的文件和文件夹