前言:
为了方便查看博客,特意申请了一个公众号,附上二维码,有兴趣的朋友可以关注,和我一起讨论学习,一起享受技术,一起成长。
1. union
在 union 中所有的数据成员共用一个空间,同一时间只能存储其中一个数据成员,所有数据成员具有相同的起始地址。
union StateMachine
{
char character;
int number;
char *str;
double exp;
};
联合里面的东西共享内存,一个 union 只配置一个足够大的空间来容纳最大长度的数据成员,如上,最大的数据长度是 double 型,所以 StateMachine 的空间大小就是 double 数据类型的大小。
2. union大小端判断
大端: 字数据的高字节存储在地地址中,而字数据的低字节则存储在高地址中;
小端: 字数据的高字节存储在高地址中,而字数据的低字节存储在低地址中。
判断程序:
void Is_BigOrLittleEndian(void)
{
union un_test
{
int i;
char ch;
char a[2];
}un,*ptr;
un.i = 0x1234;
ptr = &un;
ptr->a[0] = 0x56;
ptr->a[1] = 0x78;
printf("ptr->i = %x \r\n",ptr->i);
if (un.ch == 0x12)
{
printf("System is Big Endian \r\n");
}
else
{
printf("System is Little Endian \r\n");
}
}
注:
16 位的数据 0x1234,按小端存储如下:
地址 | 数据 |
---|---|
0x8001 | 0x12 |
0x8000 | 0x34 |
16 位的数据 0x1234,按大端存储如下:
地址 | 数据 |
---|---|
0x8001 | 0x34 |
0x8000 | 0x12 |
union 成员共享同一块大小的内存,一次只能使用其中的一个成员。如果对union 的某一个成员赋值,这会覆盖其他成员的值(前提是成员所占字节数相同,当成员所占字节数不同时只会覆盖相应字节上的值,比如对 char 成员赋值就不会把整个 int 成员覆盖掉,因为 char 只占一个字节,而 int 占四个字节);union 定义的变量的存放顺序是所有成员都从低地址开始存放的。
3. enum
enum enum_type_name
{
ENUM_CONST_1,
ENUM_CONST_2,
…
ENUM_CONST_n;
}enum_variable_name;
(1)enum_type_name 类型是对一个变量取值范围的限定,花括号内为其取值范围,即 enum_type_name 的变量 enum_variable_name 只能取花括号 内任何一个值,如果赋给该变量的值不在列表中,则会报错或警告。如下,一个单片机的 IO 端口具备多个功能,就可以用 enum 进行定义。
typedef enum
{
IO4_GPIO4, //GPIO功能
IO4_PWM4_OUT, //PWM输出
IO4_TM4_CH_IN, //定时器4输入
IO4_UART2_TX, //串口2的发送引脚
}IOM_GPIO4_CTRL;
(2)enum 变量类型还可以给其中的常量符号赋值,如果不赋值则会从被赋初值的那个常量开始依次加 1,如果都没有赋值,它们的值从 0 开始依次递增 1。如分别用一个常数表示不同颜色:
enum Color_Define
{
GREEN = 1;
RED;
BLUE;
GREEN_RED = 10;
GREEN-BLUE;
}Color_Val;
表示常量的数值为:GREEN = 1;RED = 2;BLUE = 3;GREEN_RED = 10;GREEN-BLUE = 11;
sizeof(Color_Val ) = 4:“枚举型尺寸是能够容纳最大枚举子值的整数尺寸”,“枚举类型中枚举子的值必须要能用一个int型表述”。也就是说,枚举型的尺寸不能超过 int 型,但不必等于 int 型,只要能容纳最大枚举子就行。
(3)枚举与 #define 宏的区别
a. #define 宏常量是在预编译阶段进行简单替换。枚举常量则是在编译的时候确定其值;
b. 一般在编译器里,可以调试枚举常量,但是不能调试宏常量;
c. 枚举可以一次定义大量相关的常量,而 #define 宏一次只能定义一个。