Chinaunix首页 | 论坛 | 博客
  • 博客访问: 502957
  • 博文数量: 102
  • 博客积分: 4001
  • 博客等级: 上校
  • 技术积分: 756
  • 用 户 组: 普通用户
  • 注册时间: 2008-08-21 16:01
文章分类

全部博文(102)

文章存档

2011年(1)

2010年(1)

2009年(56)

2008年(44)

我的朋友

分类: C/C++

2009-03-06 11:20:05

int i=1;  
char *p=(char *)&i;  
if(*p==1)    
    printf("1"); 
else
    printf("2");

          大小端存储问题,如果小端方式中(i占至少两个字节的长度)则i所分配的内存最小地址那个字节中就存着1,其他字节是0.大端的话则1在i的最高地址字节 处存放,char是一个字节,所以强制将char型量p指向i则p指向的一定是i的最低地址,那么就可以判断p中的值是不是1来确定是不是小端。

请写一个C函数,若处理器是Big_endian的,则返回0;若是Little_endian的,则返回1
解答:
int checkCPU( )
{
    {
           union w
           {  
                  int a;
                  char b;
           } c;
           c.a = 1;
           return(c.b ==1);
    }
}
剖析:
嵌入式系统开发者应该对Little-endian和Big-endian模式非常了解。采用Little- endian模式的CPU对操作数的存放方式是从低字节到高字节,而Big-endian模式对操作数的存放方式是从高字节到低字节。例如,16bit宽 的数0x1234在Little-endian模式CPU内存中的存放方式(假设从地址0x4000开始存放)为:
内存地址
0x4000
0x4001
存放内容
0x34
0x12
而在Big-endian模式CPU内存中的存放方式则为:
内存地址
0x4000
0x4001
存放内容
0x12
0x34
32bit宽的数0x12345678Little-endian模式CPU内存中的存放方式(假设从地址0x4000开始存放)为:
内存地址
0x4000
0x4001
0x4002
0x4003
存放内容
0x78
0x56
0x34
0x12
而在Big-endian模式CPU内存中的存放方式则为:
内存地址
0x4000
0x4001
0x4002
0x4003
存放内容
0x12
0x34
0x56
0x78
联合体union的存放顺序是所有成员都从低地址开始存放,面试者的解答利用该特性,轻松地获得了CPU对内存采用Little-endian还是Big-endian模式读写。如果谁能当场给出这个解答,那简直就是一个天才的程序员。

#include
#include

int main(void)
{
        union {
                int x;
                char y[3];
                struct x {
                        int x;
                        int y;
                }z;
        }xx;
        memset(&xx, 0, sizeof(xx));
        printf("0 %d x:%d y:%s z.x:%d z.y:%d\n", sizeof(xx), xx.x, xx.y, xx.z.x, xx.z.y);
        strcpy(xx.y, "ab");
        printf("1 %d x:%d y:%s z.x:%d z.y:%d\n", sizeof(xx), xx.x, xx.y, xx.z.x, xx.z.y);
        xx.x=100;
        printf("2 %d x:%d y:%s z.x:%d z.y:%d\n", sizeof(xx), xx.x, xx.y, xx.z.x, xx.z.y);
        xx.z.x=1;
        xx.z.y=2;
        printf("3 %d x:%d y:%s z.x:%d z.y:%d\n", sizeof(xx), xx.x, xx.y, xx.z.x, xx.z.y);
        return 0;
}


编译运行此程序:
gcc -Wall union.c
./a.out
得到如下输出结果:

0 8 x:0 y: z.x:0 z.y:0
1 8 x:25185 y:ab z.x:25185 z.y:0
2 8 x:100 y:d z.x:100 z.y:0
3 8 x:1 y: z.x:1 z.y:2

为什么会得到这个结果呢?
因为union结构是对同一块内存区域的内容按照不同数据类型要求来处理的。
比如此程序中:xx这个结构的三个成员中,最大的是z,因为有两个int所以占8个字节,因此整个结构的size取最大的成员的size,得到8。因此有8个字节的内存区域 0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00
当开始全置0时,无论如何理解,数值都是0,字符全为空。

当把y的值置为ab时,xx结构理解为y,即内存区域是这样的:0x61 0x62 0x00 0x00 0x00 0x00 0x00 0x00一个字符串, 因为a的ASCII码值为0x61,b的ASCII码值为0x62。因此此时输出xx.y的值就是ab。但如果对同样这块内存区域0x61 0x62 0x00 0x00 0x00 0x00 0x00 0x00理解成一个int时,就取前4位为一个整数,即0x00006261(我这里是little-endiam),因此此时打印zz.x值为25185,同样如果打印xx.z.x和xx.z.y的值就得到0x00006261和0x00000000,即25185和0

当把xx.x的值置为100时,即0x64,内存区域变成了0x64 0x00 0x00 0x00 0x00 0x00 0x00 0x00,按照同样的理解,如果理解为一个整数则为0x00000064,即100,如果理解为字符串,则只有一个字符d(d的ASCII值为0x64,即100),所以打印xx.y显示为d,同理zz.x=100而zz.y=0

当最后xx.z.x=1,xx.z.y=2时,内存区域变成了0x01 0x00 0x00 0x00 0x02 0x00 0x00 0x00。同理理解
阅读(1901) | 评论(1) | 转发(0) |
0

上一篇:RAID5的配置步骤

下一篇:图解fork

给主人留下些什么吧!~~

chinaunix网友2009-03-06 11:21:44

这本来是两篇文章,但是要说明第一篇文章中后面的那段代码,就得理解联合体union结构的存储顺序,第二篇文章讲的很清楚,就是这样