1.共用体union
共用体类型的定义、变量定义和使用
(1)共用体union和结构体struct在类型定义、变量定义、使用方法上很相似。
(2)union的sizeof测到的大小实际是union中各个元素里面占用内存最大的那个元素的大小。
共用体和结构体的相同和不同
(1)相同点就是操作语法几乎相同。
(2)不同点是本质上的不同。struct是多个独立元素(内存空间)打包在一起;union是一个元素(内存空间)的多种不同解析方式。
2.大小端模式
(1)大端模式(big endian)和小端模式(little endian)。
高字节对应高地址(大端模式)、高字节对应低地址(小端模式)
(1)现实的情况就是:有些CPU公司用大端(譬如C51单片机);有些CPU用小端(譬如ARM)。(大部分是用小端模式,大端模式的不算多)。
经典笔试题:用C语言写一个函数来测试当前机器的大小端模式。
用union来测试机器的大小端模式
//返回1为小端模式,返回0为大端模式
int is_little_endian(void)
{
union myunion s1;
s1.a = 1;
return s1.b;
}
指针方式来测试机器的大小端
//返回1为小端模式,返回0为大端模式
int is_little_endian2(void)
{
int a = 1;
char b = *((char *)(&a));
return b;
}
看似可行实则不行的测试大小端方式:位与、移位、强制类型转化
(1)位与运算。
结论:位与的方式无法测试机器的大小端模式。(表现就是大端机器和小端机器的&运算后的值相同的)
理论分析:位与运算是编译器提供的运算,这个运算是高于内存层次的(或者说&运算在二进制层次具有可移植性,也就是说&的时候一定是高字节&高字节,低字节&低字节,和二进制存储无关)。
(2)移位
结论:移位的方式也不能测试机器大小端。
理论分析:原因和&运算符不能测试一样,因为C语言对运算符的级别是高于二进制层次的。右移运算永远是将低字节移除,而和二进制存储时这个低字节在高位还是低位无关的。
(3)强制类型转换
同上
3.枚举
enumeration 类型定义
定义方法1,定义类型和定义变量分离开
// 定义方法1,定义类型和定义变量分离开
enum week
{
SUN, // SUN = 0
MON, // MON = 1;
TUE,
WEN,
THU,
FRI,
SAT,
};
enum week today;
定义方法2,定义类型的同时定义变量
// 定义方法2,定义类型的同时定义变量
enum week
{
SUN, // SUN = 0
MON, // MON = 1;
TUE,
WEN,
THU,
FRI,
SAT,
}today,yesterday;
定义方法3,定义类型的同时定义变量
// 定义方法3,定义类型的同时定义变量
enum
{
SUN, // SUN = 0
MON, // MON = 1;
TUE,
WEN,
THU,
FRI,
SAT,
}today,yesterday;
定义方法4,用typedef定义枚举类型别名,并在后面使用别名进行变量定义
typedef enum week
{
SUN, // SUN = 0
MON, // MON = 1;
TUE,
WEN,
THU,
FRI,
SAT,
}week;
定义方法5,用typedef定义枚举类型别名,并在后面使用别名进行变量定义
typedef enum
{
SUN, // SUN = 0
MON, // MON = 1;
TUE,
WEN,
THU,
FRI,
SAT,
}week;