【C语言】自定义类型——联合体和枚举

2024-05-06 13:12

本文主要是介绍【C语言】自定义类型——联合体和枚举,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、联合体(共用体)

1.1 联合体的声明

与结构体一样,联合体也是由一个或者多个成员构成,这些成员可以是不同的类型

看一个联合体的例子:

可以看出联合体和结构体的声明很相似,只是关键字发生了改变。

1.2 联合体的特点

大家不妨猜猜这个联合体变量的大小是多少。

答案为什么是4呢?这就与联合体的特点有关了。

联合体虽然可以有很多成员,但是所有的成员都公用一块内存空间,这样联合体变量的大小至少是最大成员的大小(确保联合体变量能存下最大的成员)。

在上面的联合体中int为4字节,char为1字节,所以un这个联合体变量为4字节就可以保存任何成员

那么我们怎么确定所有成员公用一个内存空间呢?来看下面代码:

由于联合体成员公用一块内存,这就导致一个联合体变量某一刻只能保存一个成员的值。

以上面的联合体举例:如果给un.a赋值,再给un.b赋值,那么un.a的值就被un.b的值覆盖了,此时un.a可以当做不存在了。(在练习中讲内存的具体分配)

最后整体观察一下联合体和结构体的区别:

1.3 联合体大小计算

联合体大小遵循两条规则:

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

 

联合体可以存任意成员的值,但只能存一个,那我们需要哪个成员直接创建不就好了,要联合体干嘛,不是多此一举吗?实际上使用联合体可以节省内存,我们举个例子。

假设我们要搞一个活动,需要上线一个礼品兑换单,礼品单中有三种商品:图书,杯子,衬衫。

每一种商品都有:库存量,价格,商品类型和商品类型相关信息。

图书:书名,作者,页数

杯子:设计,可选颜色

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

如果我们不加思考,也能写出一个结构体表示所有信息

这样的结构体很容易就能写出来,也很方便使用,但是这样的结构体包含了所有商品的信息,就会导致结构体的大小偏大,比较浪费内存。因为对于有些商品,有些属性不需要用到。

例如:商品为图书,就不需要design, colors, sizes三个属性。

所以,我们可以把公共属性单独拎出来,特殊属性使用联合体,能在一定程度上节省空间。

1.4 联合体的练习

再写练习之前,我们需要了解联合体成员的内存分配。

我们来看下面的代码:

我们能看到:我们对成员c赋值,但是只改变了较小地址的一个字节的值,其他内存中值不变。

首先,我们知道,联合体所有成员的地址都相同,都是联合体变量的较低地址;那么,在对联合体中成员赋值时,会遵循该成员的类型大小,类型大小是多少字节就修改多少字节内的值。同样,将联合体成员的值取出时也遵循成员类型大小,是多大就取多少字节的内容。用图来表示:

现在,我们可以使用这个来轻松完成一道面试题:判断当前机器是大端?还是小端?

这个题目似乎很熟悉,没错,在第一次谈大小端中我们写了一遍这道题,当时我们是这么写的:

现在我们可以通过联合体更容易写出这样的程序:

二、枚举

2.1 枚举的声明

枚举顾名思义就是——列举

把所有可能的值都列举出来。

比如,在我们日常生活中:

一周中的星期一到星期日是有限的7天,可以一一列举

性别:男,女,保密,可以一一列举

月份有12个月,可以一一列举

能描述的颜色是有限的,可以一一列举

这些数据的表示就可以使用枚举。

 

接下来详细讲解一下如何声明一个枚举:

2.2 枚举的使用

声明完枚举类型后,我们可以创建该枚举类型的变量:

同样,我们也可以直接使用声明了的枚举类型中的枚举常量:

注意:枚举常量不能被修改值,只能在声明时赋初值时更改初值。

2.3 枚举的优点

我们学完枚举后,知道枚举就是定义常量,常量的值默认从0开始,依次增1,也可自己修改初值

但是,#define也可以定义常量,那么我们为什么要创建枚举这种类型呢?

既然枚举这种类型,那么一定有它的优点,我们来看一下:

①增加代码的可读性和可维护性

我们举个例子:

除了这个好处,枚举还有其他优点:

②和#define定义的标识符比较,枚举有类型检查,更加严谨。

③便与调试,预处理阶段会删除#define定义的符号,调试时就无法知道该值得含义。

④使用方便,一次可以定义多个常量。

⑤枚举常量遵循作用域规则,枚举声明在函数内,只能在函数内使用,而#define不遵循,会将代码中所有的标识符都删除。

这篇关于【C语言】自定义类型——联合体和枚举的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Go语言中make和new的区别及说明

《Go语言中make和new的区别及说明》:本文主要介绍Go语言中make和new的区别及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1 概述2 new 函数2.1 功能2.2 语法2.3 初始化案例3 make 函数3.1 功能3.2 语法3.3 初始化

Go语言中nil判断的注意事项(最新推荐)

《Go语言中nil判断的注意事项(最新推荐)》本文给大家介绍Go语言中nil判断的注意事项,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1.接口变量的特殊行为2.nil的合法类型3.nil值的实用行为4.自定义类型与nil5.反射判断nil6.函数返回的

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁

C++11作用域枚举(Scoped Enums)的实现示例

《C++11作用域枚举(ScopedEnums)的实现示例》枚举类型是一种非常实用的工具,C++11标准引入了作用域枚举,也称为强类型枚举,本文主要介绍了C++11作用域枚举(ScopedEnums... 目录一、引言二、传统枚举类型的局限性2.1 命名空间污染2.2 整型提升问题2.3 类型转换问题三、C

Java实现自定义table宽高的示例代码

《Java实现自定义table宽高的示例代码》在桌面应用、管理系统乃至报表工具中,表格(JTable)作为最常用的数据展示组件,不仅承载对数据的增删改查,还需要配合布局与视觉需求,而JavaSwing... 目录一、项目背景详细介绍二、项目需求详细介绍三、相关技术详细介绍四、实现思路详细介绍五、完整实现代码

一文详解Java Stream的sorted自定义排序

《一文详解JavaStream的sorted自定义排序》Javastream中的sorted方法是用于对流中的元素进行排序的方法,它可以接受一个comparator参数,用于指定排序规则,sorte... 目录一、sorted 操作的基础原理二、自定义排序的实现方式1. Comparator 接口的 Lam

Go语言代码格式化的技巧分享

《Go语言代码格式化的技巧分享》在Go语言的开发过程中,代码格式化是一个看似细微却至关重要的环节,良好的代码格式化不仅能提升代码的可读性,还能促进团队协作,减少因代码风格差异引发的问题,Go在代码格式... 目录一、Go 语言代码格式化的重要性二、Go 语言代码格式化工具:gofmt 与 go fmt(一)

Go语言中泄漏缓冲区的问题解决

《Go语言中泄漏缓冲区的问题解决》缓冲区是一种常见的数据结构,常被用于在不同的并发单元之间传递数据,然而,若缓冲区使用不当,就可能引发泄漏缓冲区问题,本文就来介绍一下问题的解决,感兴趣的可以了解一下... 目录引言泄漏缓冲区的基本概念代码示例:泄漏缓冲区的产生项目场景:Web 服务器中的请求缓冲场景描述代码

Go语言如何判断两张图片的相似度

《Go语言如何判断两张图片的相似度》这篇文章主要为大家详细介绍了Go语言如何中实现判断两张图片的相似度的两种方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 在介绍技术细节前,我们先来看看图片对比在哪些场景下可以用得到:图片去重:自动删除重复图片,为存储空间"瘦身"。想象你是一个

如何自定义一个log适配器starter

《如何自定义一个log适配器starter》:本文主要介绍如何自定义一个log适配器starter的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录需求Starter 项目目录结构pom.XML 配置LogInitializer实现MDCInterceptor