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.联合体和结构体的比较
结构体 | 联合体 |
---|---|
struct | union |
多个成员 | 多个成员 |
每个成员都有独立的自己独立的空间 | 所有成员共用同一块内存空间 |
这样是不是更清晰了些?
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