个粉丝

问答

专栏

资料

 发布于  2008-08-08 20:16:39
采纳率 0%
128个问答
4719

union用法总结

 
定义共用体的类型变量的一般形式为:
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 )
我来回答
回答2个
时间排序
认可量排序

vforkk

0个粉丝

1

问答

0

专栏

1

资料

vforkk 2008-08-12 10:59:15
认可0
记得在cpp版里有一个帖子,就是用union实现IP地址的快速转换的!!
这个例子更为经典。。
或将文件直接拖到这里
悬赏:
E币
网盘
* 网盘链接:
* 提取码:
悬赏:
E币

Markdown 语法

  • 加粗**内容**
  • 斜体*内容*
  • 删除线~~内容~~
  • 引用> 引用内容
  • 代码`代码`
  • 代码块```编程语言↵代码```
  • 链接[链接标题](url)
  • 无序列表- 内容
  • 有序列表1. 内容
  • 缩进内容
  • 图片![alt](url)
+ 添加网盘链接/附件

Markdown 语法

  • 加粗**内容**
  • 斜体*内容*
  • 删除线~~内容~~
  • 引用> 引用内容
  • 代码`代码`
  • 代码块```编程语言↵代码```
  • 链接[链接标题](url)
  • 无序列表- 内容
  • 有序列表1. 内容
  • 缩进内容
  • 图片![alt](url)
相关问答
无更多相似问答 去提问
举报反馈

举报类型

  • 内容涉黄/赌/毒
  • 内容侵权/抄袭
  • 政治相关
  • 涉嫌广告
  • 侮辱谩骂
  • 其他

详细说明

易百纳技术社区