自定义类型:联合和枚举

文章目录


一、联合体

1.1联合体类型的声明

联合体也是由⼀个或者多个成员构成,这些成员可以不同的类型。联合体的特点是所有成员共用同一块内存空间。所以联合体也叫:共用体。

复制代码
#include <stdio.h>

//联合类型的声明
union Un
{
	char c;
	int i;
};
int main()
{
	//联合变量的定义
	union Un hao = { 0 };
	//计算连个变量的⼤⼩
	printf("%d\n", sizeof(hao));

	return 0;
}

输出的结果为:4

1.2 联合体的特点

联合的成员是共⽤同⼀块内存空间的,这样⼀个联合变量的⼤⼩,⾄少是最⼤成员的⼤⼩(因为联合⾄少得有能⼒保存最⼤的那个成员)。

复制代码
#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;
}
复制代码
#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;
}


代码1输出的三个地址⼀模⼀样,代码2的输出,我们发现将i的第4个字节的内容修改为55了。
我们仔细分析就可以画出,un的内存布局图。
(所以 i 和 c 共用同一块内存空间)

1.3 联合体大小的计算

• 联合的⼤⼩⾄少是最⼤成员的⼤⼩。
• 当最⼤成员⼤⼩不是最⼤对⻬数的整数倍的时候,就要对⻬到最⼤对⻬数的整数倍。
下面来看代码:

复制代码
#include <stdio.h>

union Un1
{
	char arr[5];//5   1 8 1
	int i;//4         4 8 4
};


union Un2
{
	short arr[7];//14    2 8 2
	int i;//4                4
};

int main()
{
	printf("%zd\n", sizeof(union Un1));
	printf("%zd\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;//尺⼨
};

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

复制代码
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.4 联合体的一个练习
  • 写⼀个程序,判断当前机器是⼤端?还是⼩端?

根据联合体的特点巧妙得结合大小端存储方式就可写出以下代码:

复制代码
include <stdio.h>

union Un
{
	char c;
	int i;
};

int main()
{
	union Un un = { 0 };
	un.i = 1;
	if (un.c == 1)
	{
		printf("小端\n");
	}
	else
	{
		printf("大端\n");
	}
	//0x 00 00 00 01
	//
	return 0;
}

二、枚举类型

2.1 枚举类型的声明

枚举的关键字:enum
枚举顾名思义就是⼀⼀列举。
把可能的取值⼀⼀列举。
比如:

复制代码
enum Day//星期
{
 Mon,
 Tues,
 Wed,
 Thur,
 Fri,
 Sat,
 Sun
};
enum Sex//性别
{
 MALE,
 FEMALE,
 SECRET
};
enum Color//颜⾊
{
 RED,
 GREEN,
 BLUE
};

以上定义的 enum Day , enum Sex , enum Color 都是枚举类型。
{}中的内容是枚举类型的可能取值,也叫 枚举常量 。
这些可能取值都是有值的,默认从0开始,依次递增1,当然在声明枚举类型的时候也可以赋初值。

2.2 枚举类型的优点

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

  1. 增加代码的可读性和可维护性
  2. 和#define定义的标识符⽐较枚举有类型检查,更加严谨;
  3. 便于调试,预处理阶段会删除 #define 定义的符号;
  4. 使⽤⽅便,⼀次可以定义多个常量;
  5. 枚举常量是遵循作⽤域规则的,枚举声明在函数内,只能在函数内使⽤。
相关推荐
浅川.252 分钟前
xtuoj 随机数
算法
shan&cen2 分钟前
Day02 集合 | 30. 串联所有单词的子串、146. LRU 缓存、811. 子域名访问计数
java·数据结构·算法·缓存
阿方.91828 分钟前
《树与二叉树详解:概念、结构及应用》
数据结构·二叉树··知识分享
NAGNIP33 分钟前
大模型微调框架之TRL
算法
麦当_35 分钟前
SwipeMultiContainer 滑动切换容器算法指南
前端·javascript·算法
橘子131 小时前
递归,搜索与回溯算法
算法
黄贵根1 小时前
C++20 基于文本文件的类对象增删查改系统
算法·c++20
max5006001 小时前
使用OmniAvatar-14B模型实现照片和文字生成视频的完整指南
图像处理·人工智能·深度学习·算法·音视频
可触的未来,发芽的智生1 小时前
追根索源-神经网络的灾难性遗忘原因
人工智能·神经网络·算法·机器学习·架构
靠近彗星2 小时前
2.2顺序表
数据结构