一、联合的定义
联合也叫共用体。
联合也是一种特殊的自定义类型 这种类型定义的变量也包含一系列的成员,特征是这些成员公用同一块空间(所以联合也叫共用体)。比如:
//联合类型的声明
union Un
{
char c;
int i;
};
//联合变量的定义
union Un un;
//计算连个变量的大小
printf("%d\n", sizeof(un));
我们发现,这里不管是u的地址,还是u的成员变量的地址,都是一致的。
二、联合的特点
联合的成员是共用同一块内存空间的,这样一个联合变量的大小,至少是最大成员的大小(因为联合至少得有能力保存最大的那个成员)。
union Un
{
int i;
char c;
};
union Un un;
// 下面输出的结果是一样的吗?
printf("%d\n", &(un.i));
printf("%d\n", &(un.c));
//下面输出的结果是什么?
un.i = 0x11223344;
un.c = 0x55;
printf("%x\n", un.i);
结果是11223355.
说明在使用联合的时候,里面的成员变量同一时刻只能使用一个。
面试题:判断当前计算机的大小端存储。
判断大小端:只需要看第一个字节里放的是什么。
首先拿到地址,再把地址强制转换成字符指针,
*(char*)&a,如果是1,就是小端;是0,就是大端。
那么用联合体如何实现?
char c;int i;
正好c不就占了一个字节吗?
我们可以对i赋值,
然后看c里放的是什么。
int check_sys() { union Un { char c; int i; }u; u.i = 1; return u.c; }
三、联合大小的计算
联合的大小至少是最大成员的大小。
当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍。
比如:
union Un1
{
char c[5];
int i;
};
union Un2
{
short c[7];
int i;
};
//下面输出的结果是什么?
printf("%d\n", sizeof(union Un1));
printf("%d\n", sizeof(union Un2));
对于Un1,
char c[5]的每个元素大小是1,默认对齐数是8,所以他的对齐数是1;
int i大小是4,默认对齐数是8,所以他的对齐数是4.
综上,Un1的大小是最大对齐数的整数倍,也就是4的倍数。
再看成员:
char c[5]的大小是5,也是最大成员大小,
但是5并不是最大对齐数的整数倍,
所以要对齐到最大对齐数的整数倍,也就是4n,
所以最终应该是8,占8个字节。