如何讀取float型態資料在記憶體中的值

 1 #include<stdio.h>
 2 #include<stdlib.h>
 3 main()
 4 {
 5 float x;
 6 
 7 x=-1;
 8 int i;
 9 for (i=3;i>=0;i--)
10 {
11 printf("%x",((unsigned char*)&x)[i]);
12 }
13 
14 }

 或者

int i;
union {
  float num;
  unsigned char n[4];
  } q;

q.num=-1;
for(i=3;i>=0;i--)
  printf("%x ", q.n[i]);


可以得到一樣的結果
IEEE 754:
-1 = -1.0*2^+127 = 1,0111 1111,0000...23個
所以結果: 1,011=B, 1111=F, 1000=8, 0000....=0,00,00 ==>BF800000

---

1
2
3
4
5
6
7
8
9
#include <stdio.h>
int main() 
{
    float a = 12.5;
    printf("%d\n", a);
    printf("%d\n", (int)a);
    printf("%d\n", *(int *)&a);
    return 0; 
}

参考答案
该项程序输出如下所示,
0
12
1095237632
原因是:浮点数是4个字节,12.5f 转成二进制是:01000001010010000000000000000000,十六进制是:0×41480000,十进制是:1095237632。所以,第二和第三个输出相信大家也知道是为什么了。而对于第一个,为什么会输出0,我们需要了解一下float和double的内存布局,如下:

  • float: 1位符号位(s)、8位指数(e),23位尾数(m,共32位)
  • double: 1位符号位(s)、11位指数(e),52位尾数(m,共64位)

然后,我们还需要了解一下printf由于类型不匹配,所以,会把float直接转成double,注意,12.5的float和double的内存二进制完全不一样。别忘了在x86芯片下使用是的反字节序,高位字节和低位字位要反过来。所以:

  • float版:0×41480000 (在内存中是:00 00 48 41)
  • double版:0×4029000000000000 (在内存中是:00 00 00 00 00 00 29 40)

而我们的%d要求是一个4字节的int,对于double的内存布局,我们可以看到前四个字节是00,所以输出自然是0了。

这个示例向我们说明printf并不是类型安全的,这就是为什么C++要引如cout的原因了

posted @ 2012-12-16 19:17  jeremyatchina  阅读(237)  评论(0编辑  收藏  举报