定义共用体的类型变量的一般形式为: union 共用体名 { 成员列表; }变量列表; 例如: union data { int i; char c; double d; }; union data a; 共用体变量a中的成员i、c、d三个变量在内存中从同一个地址开始存储,即在同一个内存中可以用来存放几种不同类型的数据; 例如进行如下赋值: a.i = 100; a.c = 'c'; 那么此时共用体变量a中的成员i已经没有值了,因为存储该值的内存现在已经被用来存储成员c的值了。 曾经面试的时候一个面试官给我出了这么一个题目(前提是32位机): typedef union _u_ { unsigned int a; unsigned char b[2]; }u; int main(void) { i.a = 0x1234; printf("%d,%d/n",b[0],b[1]); return 0; } 如果你清楚了共用体的概念,这个题目是很简单的,但是有一点我们需要考虑的是cpu存放整形和长整形数据有两种不同的顺序,对于整型、长整型等数据类型,Big endian 认为第一个字节是最高位字节(按照从低地址到高地址的顺序存放数据的高位字节到低位字节);而 Little endian则相反,它认为第一个字节是最低位字节(按照从低地址到高地址的顺序存放数据的低位字节到高位字节),所以这个地方的值有两种可能了;对于是 Little endian型的值是 52,18;对于是Big endian型的值是18,52。 而由此延伸出来我们可以测试cpu是那种存放数据类型的; 我们知道,union 的成员本身就被存放在相同的内存空间(共享内存,正是 union 的作用),因此,我们可以将一个 char 数据和一个整型数据同时作为一个 union 的成员,得出 如下答案: int checkCPU() { { union w { int a; char b; } c; c.a = 1; return (c.b == 1); } } 实现同样的功能,我们来看看 Linux操作系统中相关的源代码是怎么做的: static union { char c[4]; unsigned long l; } endian_test = { { 'l', '?', '?', 'b' } }; #define ENDIANNESS ((char)endian_test.l) Linux 的内核作者们仅仅用一个 union 变量和一个简单的宏定义就实现了一大段代码同样的功能!由以上一段代码我们可以深刻领会到 Linux 源代码的精妙之处!(如果 ENDIANNESS=’l’表示系统为 little endian,为’b’表示 big endian )
union的用法
最新推荐文章于 2024-07-03 10:26:58 发布