前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >【C语言基础】:自定义类型(二) -->联合和枚举

【C语言基础】:自定义类型(二) -->联合和枚举

作者头像
用户11062259
发布2024-04-10 15:22:11
1080
发布2024-04-10 15:22:11
举报
文章被收录于专栏:C语言基础C语言基础

书山有路勤为径,学海无涯苦作舟。 创作不易,宝子们!如果这篇文章对你们有帮助的话,别忘了给个免费的赞哟~

一、联合体
1.1 联合体类型的声明

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

给联合体其中一个成员赋值,其他成员的值也跟着变化。

代码语言:javascript
复制
// 联合体的声明
union Un
{
	char c1;
	int i;
};
代码语言:javascript
复制
#include<stdio.h>
int main()
{
	union Un u = { 0 };  // 联合变量的定义
	printf("%zd\n", sizeof(u));  // 计算联合变量的大小
	return 0;
}

从运行结果来看:这个联合体中有两个成员变量,一个整形和一个字符型,按理说应该是5个字节的大小,但这个联合体的大小却只有4个字节,这也就是联合体的特点了。

1.2 联合体的特点

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

【代码1】

代码语言:javascript
复制
#include<stdio.h>
union Un
{
	char c1;
	int i;
};

int main()
{
	union Un u = { 0 };
	printf("%zd\n", sizeof(u));
	 printf("%p\n", &u);
	 printf("%p\n", &(u.c1));
	 printf("%p\n", &(u.i));
	return 0;
}

【代码2】

代码语言:javascript
复制
#include<stdio.h>
union Un
{
	char c;
	int i;
};

int main()
{
	union Un u = { 0 };
	u.i = 0x11223344;
	u.c = 0x55;
	printf("%x\n", u.i);
	return 0;
}

代码1输出的三个地址一模一样,代码2的输出,我们发现将i的第4个字节的内容修改为55了。 我们仔细分析就可以画出,un的内存布局图。

【结论】:联合体的成员是共用同一块内存空间的,修改一个其他的也会被修改。

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

我们再对比一下相同成员的结构体和联合体的内存布局情况。

【结构体】

代码语言:javascript
复制
struct S
{
	char c;
	int i;
};
struct S s = { 0 };

【联合体】

代码语言:javascript
复制
union Un
{
	char c;
	int i;
};
union Un un = { 0 };
1.4 联合体大小的计算
  • 联合的大小至少是最大成员的大小。
  • 当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍。
代码语言:javascript
复制
#include<stdio.h>
union Un
{
	char c[5];
	int i;
};

int main()
{
	union Un u = { 0 };
	printf("%zd\n", sizeof(u));
	return 0;
}

以这题为例,这个联合体中最大的成员是char c[5],但这个联合体的大小却并不是5个字节的大小,而是8个字节,这说明联合体的大小不全是最大成员的大小。当最大成员的大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍。这里char c[5] 就相当于5个char 类型,也就是1个字节,VS的默认值是8,所以对齐数是1,int占4个字节,所以对齐数就是4,所以这个联合体的最大对齐数就是4,但5并不是4的倍数,即要浪费3个字节变为8个字节。

【结论】:联合体的大小也要是最大对齐数的倍数。

使用联合体是可以节省空间的,举例: 比如,我们要搞一个活动,要上线一个礼品兑换单,礼品兑换单中有三种商品:图书、杯子、衬衫。 每一种商品都有:库存量、价格、商品类型和商品类型相关的其他信息。 | 图书:书名、作者、页数 | 杯子:设计 | 衬衫:设计、可选颜色、可选尺寸

这题可以直接用结构体写:

代码语言:javascript
复制
struct gift_list
{
	// 公共属性
	int stock_number;  // 库存量
	double price;  // 价格
	int item_type;  // 商品类型
	// 特殊属性
	char title[20];  // 书名
	char author[20];  // 作者
	int num_pages;  // 页数

	char desgin[30];  // 设计
	int colors;  // 颜色
	int sizes;  // 尺寸
};

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

代码语言:javascript
复制
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 desgin[30];  // 设计
		}mug;
		struct
		{
			char desgin[30];  // 设计
			int colors;  // 颜色
			int sizes;  // 尺寸
		}shirt;
	}item;
};
1.5 联合体练习

写⼀个程序,判断当前机器是大端?还是小端?

代码语言:javascript
复制
#include<stdio.h>
int check_sys()
{
	union Un
	{
		int i;
		char c;
	}u;
	u.i = 1;
	return u.c;
}

int main()
{
	int ret = check_sys();
	if (ret == 1)
		printf("小端\n");
	else
		printf("大端\n");
	return 0;
}
二、枚举类型
2.1 枚举类型的声明

枚举顾名思义就是一一列举。 把可能的取值一一列举。 比如我们现实生活中: | 一周的星期一到星期日是有限的7天,可以一一列举 | 性别有:男、女、保密,也可以一一列举 | 月份有12个月,也可以一一列举 | 三原色,也是可以意义列举 这些数据的表示就可以使用枚举了。

代码语言:javascript
复制
enum Day  // 星期
{
	Mon,
	Tues,
	Wed,
	Thur,
	Fri,
	Sat,
	Sun
};

enum Sex  // 性别
{
	MALE,
	FEMALE,
	SECRET
};

enum Color // 三原色
{
	RED,
	GREEN,
	BLUE
};

以上定义的 enum Dayenum Sexenum Color 都是枚举类型{} 中的内容是枚举类型的可能取值,也叫枚举常量

这些可能取值都是有值的,默认从0开始,依次递增1,当然在声明枚举类型的时候也可以赋初值。

代码语言:javascript
复制
enum Color // 三原色
{
	RED,
	GREEN,
	BLUE
};

int main()
{
	enum Color color = RED;
	return 0;
}

枚举常量也是有值的。

代码语言:javascript
复制
#include<stdio.h>
enum Color // 三原色
{
	RED,
	GREEN,
	BLUE
};

int main()
{
	printf("%d\n", RED);
	printf("%d\n", GREEN);
	printf("%d\n", BLUE);
	return 0;
}

默认值从0开始,往后依次加一,也可以在声明枚举类型的时候也可以赋初值。

代码语言:javascript
复制
enum Color // 三原色
{
	RED = 5,
	GREEN,
	BLUE
};
代码语言:javascript
复制
enum Color // 三原色
{
	RED,
	GREEN = 5,
	BLUE
};
2.2 枚举的优点

为什么使用枚举?

学到这里,我们会发现枚举常量和 #define 有点类似,那为什么不直接用 #define 呢?

代码语言:javascript
复制
enum Color // 三原色
{
	RED,// 0
	GREEN,// 1
	BLUE// 2
};

#define RED 0
#define GREEN 1
#define BLUE 2

枚举的优点:

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

【示例】:计算器

代码语言:javascript
复制
enum Option
{
	EXIT,
	ADD,
	SUB,
	MUL,
	DIV
};

int mian()
{
	int input = 0;
	printf("请选择:>");
	scanf("%d", &input);
	switch (input)
	{
	case ADD:  // 加法
		break;
	case SUB:  // 减法
		break;
	case MUL:  // 乘法
		break;
	case DIV:  // 除法
		break;
	case EXIT:  // 退出
		break;
	default:
		break;
	}
	return 0;
}

这样跟有利于我们代码的可读性。 注意:C语言中可以利用整数给枚举变量赋值,但是C++不可以,这是因为C++的类型检查比较严格。

代码语言:javascript
复制
enum Color // 三原色
{
	RED,// 0
	GREEN,// 1
	BLUE// 2
};

int main()
{
	// enum Color color = RED; // TRUE 
	enum Color color = 0;
	return 0;
}

enum Color color = 0; 这一句虽然在C语言中不会报错,但是在C++中会显示类型不同,无法赋值,就是因为C++的类型检查比较严格。

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2024-04-10,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 一、联合体
    • 1.1 联合体类型的声明
      • 1.2 联合体的特点
        • 1.3 相同成员的结构体和联合体对比
          • 1.4 联合体大小的计算
            • 1.5 联合体练习
            • 二、枚举类型
              • 2.1 枚举类型的声明
                • 2.2 枚举的优点
                领券
                问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档