C语言:深入了解(联合体和枚举)

avatar
作者
筋斗云
阅读量:2

目录

联合体

联合体的类型的声明

联合体的特点

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

联合体大小的计算

联合体的使用举例

联合体的类型:判断联合体是大端还是小端

枚举类型

枚举类型声明

枚举类型的优点

枚举类型的使用


联合体

联合体的类型的声明

像结构体⼀样,联合体也是由⼀个或者多个成员构成,这些成员可以不同的类型。
但是编译器只为最⼤的成员分配⾜够的内存空间。联合体的特点是所有成员共⽤同⼀块内存空间。所
以联合体也叫:共⽤体。
给联合体其中⼀个成员赋值,其他成员的值也跟着变化。

联合体输出的结果为什么是4呢?


联合体的特点

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

联合体是共⽤同⼀块内存空间的所以不能同时使用联合体,不能同时使用

我们可以看到3个地址都是一样的。

union a { 	char a; 	int b; };  int main() { 	union a p = { 0 }; 	printf("%zd\n", sizeof(union a)); 	printf("%p\n", &p); 	printf("%p\n", &p.a); 	printf("%p\n", &p.b);  } 

相同成员的结构体和联合体对比
struct bp//结构体 { 	char a; 	int b; }; 
union ar//联合体 { 	char a; 	int b; };


联合体大小的计算

联合的⼤⼩⾄少是最⼤成员的⼤⼩。
当最⼤成员⼤⼩不是最⼤对⻬数的整数倍的时候,就要对⻬到最⼤对⻬数的整数倍。


联合体的使用举例

⽐如,我们要搞⼀个活动,要上线⼀个礼品兑换单,礼品兑换单中有三种商品:图书、杯⼦、衬衫。
每⼀种商品都有:库存量、价格、商品类型和商品类型相关的其他信息

图书:书名、作者、⻚数
杯⼦:设计
衬衫:设计、可选颜⾊、可选尺⼨


下面这代码,当我们要描述一个杯⼦的时候,只用到公共属性和设计属性其他属性都没用这样非常浪费空间

struct lx { 	//公共属性 	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。?
所以我们就可以把公共属性单独写出来,剩余属于各种商品本⾝的属性使⽤联合体起来,这样就可以
介绍所需的内存空间,⼀定程度上节省了内存。

下面这联合体在同一时间只能使用一次结构体,不能同时使用2个

struct lx { 	//公共属性 	int stock_number; //库存量 	double price; //定价 	int item_type; //商品类型  	//特殊属性 	union //不完全声明只能用一次 	{ 		//同一时间只能用一次结构体,不能同时用2个  		struct //不完全声明只能用一次 		{ 			char title[20]; //书名 			char author[20]; //作者 			int num_pages; //⻚数 		}shu;  		struct //只能用一次 		{ 			char design[30]; //设计 		}bei;  		struct //只能用一次 		{ 			int colors; //颜⾊ 			int sizes; //尺⼨ 		}chens;  	}tssx;  };

联合体的类型:判断联合体是大端还是小端

把1赋值给a,在内存中是小端存放的所以用char类型的来进行判断第一个字节是不是大端小端

union a { 	int a; 	char b;  };  int main() { 	union a p = { 0 }; 	p.a = 1;// 小端:01 00 00 00 	if (p.b == 1)//判断第1个字节是不是1 	{ 		printf("小端\n"); 	} 	else 	{ 		printf("大端\n"); 	} }

枚举类型

枚举类型声明

枚举顾名思义就是⼀⼀列举。
把可能的取值⼀⼀列举。
⽐如我们现实⽣活中:
⼀周的星期⼀到星期⽇是有限的7天,可以⼀⼀列举?
性别有:男、⼥、保密,也可以⼀⼀列举
⽉份有12个⽉,也可以⼀⼀列举?
三原⾊,也是可以意义列举

enum a//星期 { 	z1 = 1,//周1 	z2,//周2 	z3,//周3 	z4,//周4 	z5,//周5 	z6,//周6 	z7 //周7 };  enum r//性别 { 	nan, 	nv, 	bm };  enum e//颜色 { 	RED, 	GREEN, 	BLUE }; 

下面这枚举我们可以看到不赋值的话,是从0开始的


赋值的话可以打印赋值的数值


给其中一个赋值的话后面的都会递增1


枚举类型的优点

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


枚举类型的使用
//打印菜单 cd() { 	printf("***************************\n"); 	printf("****** 1.加法   2.减法 ****\n"); 	printf("****** 3.除法   4.乘法 ****\n"); 	printf("************ 0.退出 *******\n"); } //枚举 enum p { 	tc,//0 	jian,//1 	jia,//2 	chu,//3 	cheng//4 }; jian_ys(int a,int b)//加法 { 	int r = a + b; 	printf("%d\n", r); } jia_ys(int a, int b)//减法 { 	int r = a - b; 	printf("%d\n", r); } chu_ys(int a, int b)//除法 { 	int r = a / b; 	printf("%d\n", r); } cheng_ys(int a, int b)//乘法 { 	int r = a * b; 	printf("%d\n", r); } int main() { 	int a = 0; 	int b = 0; 	int r = 0; 	do 	{ 		//打印菜单 		cd(); 		scanf("%d", &r); 		switch (r) 		{ 		case tc: 			printf("退出\n"); 			break; 		case jian://1 			printf("请输入2个数值:\n"); 			scanf("%d %d", &a, &b); 			jian_ys(a,b); 			break;//2 		case jia: 			printf("请输入2个数值:\n"); 			scanf("%d %d", &a, &b); 			jia_ys(a, b); 			break; 		case chu://3 			printf("请输入2个数值:\n"); 			scanf("%d %d", &a, &b); 			chu_ys(a, b); 			break; 		case cheng://4 			printf("请输入2个数值:\n"); 			scanf("%d %d", &a, &b); 			cheng_ys(a, b); 			break; 		default: 			printf("输入错误请重新输入\n"); 			break; 		} 	} while (r); }

广告一刻

为您即时展示最新活动产品广告消息,让您随时掌握产品活动新动态!