C · 进阶 | 联合体?看一遍就够了

CSDN话题挑战赛第2期
参赛话题:学习笔记


在这里插入图片描述
啊我摔倒了..有没有人扶我起来学习....



前言

结构体、枚举、联合体都是自定义类型,结构体主要知识点结构体内存对齐可参考《C | 结构体内存对齐》枚举请参考《C | 枚举?看一遍就够了》


联合体

1. 联合体的定义

  • 与结构体非常类似
union Un
{
	char c;
	int i;
	double d;
};

2. 联合体的内存大小

  • 联合体所占内存又是多大呢?
int main()
{
	union Un un;
	printf("%d\n", sizeof(union Un));
	printf("%d\n", sizeof(un));

	return 0;
}

打印结果:在这里插入图片描述可以看出来,是占用内存最大的成员double决定的8个字节。真的是这样吗?

  • 结构体又叫做共同体,顾名思义,它的成员都共用一个内存空间
int main()
{
	union Un un;
	printf("%p\n", &un);
	printf("%p\n", &(un.c));
	printf("%p\n", &(un.i));
	printf("%p\n", &(un.d));

	return 0;
}

打印结果:在这里插入图片描述可以看到,确实是在同一个空间中。

  • 看起来联合体的内存就是成员中占用内存最大的那个决定,带着这个问题思考下述代码
在这里插入代码片

3. 联合体应用场景

  • 看完以上内容,铁汁们可能会觉得联合体根本没鸟用嘛!成员之间共同内存不就会互相覆盖了嘛!别急,咱们细细研究一下~
  • 我们设想一种情况,比如你是学生,有你的学号,储存在联合体中。将来有一天,你变成了老师,学号已经用不着了,取而代之的是职工号,这时联合体的作用不就出来啦~可以认为联合体的作用就是节省内存
  • 再来一个巧妙的用法!铁汁们还记得判断大小端字节序的方法吗,今天咱们利用联合体试试看
  1. 我们先用暴力的方法看看我们VS2019的编译器是什么字节序,先来段代码
int main()
{
	int num = 0x11223344;
	return 0;
}

然后开始调试,打开内存窗口看看。不会调试的铁汁可以看看《C | 实用调试技巧》
在这里插入图片描述可以看到是小端字节序存储方式

  1. 如果直接用代码能不能测试出来呢?先来段以前的方法
int main()
{
	int num = 1;
	char* p = (char*)#//int*

	if (*p == 1)
		printf("小端\n");
	else
		printf("大端\n");

	return 0;
}

打印结果:在这里插入图片描述

在这里插入图片描述
3. 接下来用联合体试试看

int check_sys()
{
	union Un
	{
		char c;
		int i;
	}u;
	u.i = 1;
	return u.c;
}

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

打印结果:在这里插入图片描述

分析分析:
首先把1赋给ii是整型,占4个字节。然后用打印cc是字符型,占1个字节。而i又和c共用内存,小端存储的话1就位于低地址处,于是打印c就会是1
在这里插入图片描述

4. 联合体内存误区

  • 还是得谈谈联合体的内存,我们上述计算过一次联合体的内存
union Un
{
	char c;
	int i;
	double d;
};

在这里插入图片描述

  • 可是,真的取决于最大成员double的大小吗?我们来看看这段代码
union Un
{
	char arr[5];
	int i;
};
int main()
{
	printf("%d\n", sizeof(union Un));

	return 0;
}

打印结果:在这里插入图片描述
可以看出并不是arr[5]所决定的5个字节

  • 所以,其实联合体也是需要对齐的,规则很简单,就是:
    1. 联合体的大小至少是最大成员的大小
    2. 当最大成员大小不是最大对齐数的整数倍时,就要对齐到最大对齐数的整数倍(注意,char arr[5]虽然是数组,但是它的对齐数取决于类型char,所以它的对齐数是1

总结

  • 联合体的用处和特点想必铁汁们都充分了解啦~如果铁汁对于内存对齐不够了解的话,还是需要学习一下《C | 结构体内存对齐》

在这里插入图片描述

  • 32
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 31
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

CGod

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值