联合体
联合体类型的声明
像结构体⼀样,联合体也是由一个或者多个成员构成,这些成员可以不同的类型。但是编译器只为最
大的成员分配足够的内存空间。联合体的特点是所有成员共用同⼀块内存空间。所以联合体也叫:共
⽤体。
给联合体其中一个成员赋值,其他成员的值也跟着变化。
#include <stdio.h>
union Un
{
char c;
int i;
};
int main()
{
union Un un = { 0 };
printf("%d\n", sizeof(un));
return 0;
}
联合体的特点
联合的成员是共⽤同一块内存空间的,这样⼀个联合变量的大小,至少是最大成员的大小(因为联合⾄少得有能力保存最大的那个成员)。
#include <stdio.h>
union Un
{
char c;
int i;
};
int main()
{
union Un un = { 0 };
printf("%p\n", &(un.i));
printf("%p\n", &(un.c));
printf("%p\n", &un);
return 0;
}
这就充分说明联合体是共用一个空间的。
#include <stdio.h>
union Un
{
char c;
int i;
};
int main()
{
union Un un = { 0 };
un.i = 0x11223344;
un.c = 0x55;
printf("%x\n", un.i);
return 0;
}
相同成员的结构体和联合体对比
struct S
{
char c;
int i;
};
struct S s = { 0 };
union Un
{
char c;
int i;
};
union Un un = { 0 };
联合体大小的计算
- 联合的大小至少是最大成员的大小。
- 当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍。
#include <stdio.h>
union Un1
{
char c[5];
int i;
};
//c占了5个字节,不是最大对齐数4的倍数,因此union Un1的大小为8
union Un2
{
short c[7];
int i;
};
//c占了14个字节,不是最大对齐数4的倍数,因此union Un2的大小为16
int main()
{
printf("%d\n", sizeof(union Un1));
printf("%d\n", sizeof(union Un2));
return 0;
}
练习
写⼀个程序,判断当前机器是大端,还是小端。
#include<stdio.h>
int check_sys()
{
union
{
int i;
char c;
}un;
un.i = 1;
return un.c;
}
int main()
{
int ret = check_sys();
if (ret == 1)
{
printf("小端\n");
}
else
{
printf("大端\n");
}
return 0;
}
枚举类型
枚举类型的声明
enum Day
{
Mon,//枚举常量
Tues,
Wed,
Thur,
Fri,
Sat,
Sun
};
enum Sex
{
MALE,
FEMALE,
SECRET
};
enum Color
{
RED,
GREEN,
BLUE
};
以上定义的 enum Day
, enum Sex
, enum Color
都是枚举类型。
{}
中的内容是枚举类型的可能取值,也叫枚举常量。
这些可能取值都是有值的,默认从0开始,依次递增1,当然在声明枚举类型的时候也可以赋初值。
enum Color
{
RED=2,
GREEN=4,
BLUE=8
};
枚举类型的优点
- 增加代码的可读性和可维护性
- 和
#define
定义的标识符⽐较枚举有类型检查,更加严谨。 - 便于调试,预处理阶段会删除
#define
定义的符号 - 使用方便,一次可以定义多个常量
- 枚举常量是遵循作用域规则的,枚举声明在函数内,只能在函数内使用
枚举类型的使用
enum Color
{
RED = 1,
GREEN = 2,
BLUE = 4
};
int main()
{
enum Color clr = GREEN;
printf("%d\n", clr);
clr = 5;
printf("%d\n",clr);
return 0;
}
在C语言中是可以拿整数给枚举变量赋值,但是在C++是不行的,C++的类型检查比较严格。