前言
欢迎大家来到我的博客!!!
今天来讲下C语言的另位两种自定义数据类型—联合体。
联合体
联合体和结构体一样都由多个成员变量构成,并且这些变量可以为不同类型。
但是编译器在给联合体分配空间时,只为成员变量中最大变量类型分配了的足够的内存大小,其他成员共用同一块内存空间,因此联合体又别名共用体。
至于联合体的声明与结构体相类似:
union UN//联合体声明
{
int i;
short s;
char c;
};
结合前面对联合体的定义,我们可以看出这个联合体类型的大小为最大的int类型,也就是为4字节。但其内存布局又是如何呢?以下是使用VS22显示的该联合体的内存布局情况:
可见联合体A创建的内存空间为一个int类型的大小,而char类型c与short类型s与i共用同一块内存空间,并且c与i都从0位置开始对齐。
联合体的大小至少为最大成员变量的类型的大小,但当最大成员对齐数的大小不是最大对齐数的整数倍时,这便需要用到对齐规则的知识了。所以此时联合体按最大对齐数的整数倍对齐。
然后根据上面对于联合体的定义可以猜测以下代码的结果:
#include <stdio.h>
int main()
{
//联合变量的定义
union UN un = {0};
un.i = 0x11223344;
un.c = 0x55;
printf("%x\n", un.i);
return 0;
}
根据un的内存布局情况我们可以看出c将i的低地址处44部分替换为了55:
可见联合体的使用可能会覆盖先前的数据,因此,使用联合体最后不要同时使用。
然后,我们其实也可以使用联合体判断先前博客【C语言的内存函数以及数据在内存中的存储方式】的大小端字节序问题:
#include <stdio.h>
int system_check()
{
union UN
{
int i
char c;
}un;
un.i = 1;
return un.c;
}
int main()
{
int ret = system_check();
if (ret == 1)//un中的c为1说明为小端
printf("系统为小端!\n");
else
printf("系统为大端!\n");
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;//尺⼨
};
但是,不同礼品拥有特殊属性,如果我们每个礼品都像上述结构体一样声明,每样礼品都存在未使用的特殊属性,比如定义图书的礼品时,没有使用到design这个特殊属性,而design占30字节,这样不就浪费了大量的内存空间。
因此,我们其实可以使用联合体如是声明:
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;
};
由于我们在定义一个礼品时,比如定义图书,就用不到杯子和衬衫的特殊属性。于是我们可以将不同的礼品的特殊属性用联合体表示,这样当我们定义不同礼品时既可以选择到该礼品的特殊属性,又可以不至于浪费大量的内存空间,因为此时该结构体在特殊属性的大小只为最大的结构体类型也就是book的大小。
总结
以上便是本人有关联合体的自定义数据类型的理解,如有出错,欢迎大佬指正!!