【自定义类型:联合体】

1.前言:引入

Tips:阅读本文之前可以先去了解一下结构体,两种有诸多相似之处,可以帮助理解~
链接: 自定义类型:结构体

union Un  //声明
{
	char c;//1
	int i; //4
};

int main()
{
	union Un u = {0};  //初始化
	printf("%zd\n", sizeof(u));//4

	return 0;
}

这一段代码的答案是?

在这里插入图片描述
这似乎与结构体的计算方法有所不同。接下来让我们深入了解这个有些陌生的类型。

2.联合体类型的声明

声明与初始化与结构体并无太大区别

union Un  //声明
{
	char c;//1
	int i; //4
};

int main()
{
	union Un u = { 0 };  //初始化
	return 0;
}

3.联合体的特点

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

例一:

union Un  //声明
{
	char c;//1
	int i; //4
};

int main()
{
	union Un u = {0};  //初始化
	printf("%zd\n", sizeof(u));//4

	printf("%p\n", &u);
	printf("%p\n", &(u.i));
	printf("%p\n", &(u.c));

	return 0;
}

尝试寻找联合体中不同成员的地址,结果是
在这里插入图片描述
说明联合体里的成员共用一块空间。

例二:

union Un
{
	char c;//1
	int i; //4
};

int main()
{
	union Un u = { 0 };
	u.i = 0x11223344;
	u.c = 0x55;

	return 0;
}

在这里插入图片描述
自行分析,结果如图所示

4.联合体和结构体的比较

结构体联合体
structunion
多个成员多个成员
每个成员都有独立的自己独立的空间所有成员共用同一块内存空间

在这里插入图片描述
这样是不是更清晰了些?
PS:联合体也是有匿名的,与结构体用法一致。

5.联合体大小的计算

• 联合的大小至少是最大成员的大小。
• 当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍。
对齐数 = 编译器默认的⼀个对齐数 与 该成员变量大小的较小值。
VS 中默认的值为 8
Linux中 gcc 没有默认对齐数,对齐数就是成员自身的大小

详细的可以参考结构体的计算方式
链接: 自定义类型:结构体

例一:

union Un
{
	char c[5];//5 char为1 默认为8  取1
	int i; //4  int为4 默认为8 取4
};
int main()
{
	union Un u = { 0 };
	printf("%zd\n", sizeof(u));//必须是4的倍数,4不够大取8,答案为8
	return 0;
}

例二:

int main()
{
	union Un u = { 0 };
	printf("%zd\n", sizeof(u));//必须是4的倍数,要存5(大的)取8
	return 0;
}


union Un
{
	short s[7];//14  short为2 默认为8 取2
	int i; //4 int为4 默认为8 取4
};

int main()
{
	union Un u = { 0 };
	printf("%zd\n", sizeof(u)); //必须是4的倍数 要存14(大的)取16,答案为16
	return 0;
}

例三:(回顾)

union Un  //声明
{
	char c;//1 char为1 默认为8 取1
	int i; //4  int为4 默认为8 取4
};

int main()
{
	union Un u = { 0 };  
	printf("%zd\n", sizeof(u))//必须是4的倍数 要存4(大的)取4,答案为4
	return 0;
}

这就是开头的题目,到现在你是不是已经明白了原因了呢~

6.联合体的应用

应用一:

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

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

对于库存量,价格,不同的商品都有一致的形式,但商品类型的相关信息却不尽相同。
如果是用结构体实现,我们应该将所有所需量存入。

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;
};

int main()
{
    struct gift_list gl;
    
    return 0;
}

这样,因为联合体的特性,三种特殊属性共用一块空间。每次只要找需要的即可(比如gl.item.book之类的),在一定程度上节约空间。

应用二:写一个程序,判断当前机器是大端?还是小端?

int check_sys()
{
	int n = 1;
	return *(char*)&n;
}

int main()
{
	int ret = check_sys();//小端:返回1,大端:返回0
	if (ret == 1)
		printf("小端\n");
	else
		printf("大端\n");
	return 0;
}
  • 用联合体的版本
int check_sys()
{
	union Un
	{
		char c;
		int i;
	}u;
	u.i = 1;
	return u.c;//小端:返回1,大端:返回0
}

int main()
{
	int ret = check_sys();//小端:返回1,大端:返回0
	if (ret == 1)
		printf("小端\n");
	else
		printf("大端\n");
	return 0;
}

7.结语

希望能有些帮助,有不对的地方多多指出0.0

  • 19
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 4
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值