随笔 - 147  文章 - 5  评论 - 6  阅读 - 81298

printf以"%d"输出浮点数(转)

http://blog.chinaunix.net/u3/104010/showart_2275754.html
曾看到printf的一道题,挺有意思,记录一下。

float value = 1.0;
printf("value_int = %d\n", value);

printf("value_float = %f\n", value);


应该输出什么?乍看这个题,很简单,浮点数1.0在内存中的存储形式是0x3f800000。float型在内存中占4Byte, int型也占4字节,按说就直接输出0x3f800000的十进制形式就可以呗。
谁知道一运行大跌眼镜,打印信息:
value_int = 0
value_float = 1.0
怎么回事?
使用gcc -S的参数把.c程序变成.s的汇编语言程序。我们可以看到:

flds    -8(%ebp)
fstpl    4(%esp)
movl    $.LC1, (%esp)
call    printf


其中value的值被存在-8(�p)处,.LC1处存储"value = %d\n"字符串。
flds 指令意为把单精度value的值放入FPU的st7寄存器(64bit)中,此时st7中的值为0x3f80000000000000
接着fstpl 指令把FPU的寄存器中的值以双精度的形式出栈,并存储在4(%esp)处。即(%esp+4)中的值为0x00000000, (%esp+8)值为0x3f800000. 
调用printf时,由于指定打印方式是%d,故printf只读取(%esp+4)的四个字节并把它们解释为十进制整形--0,而不会顾及到(%esp+8)的正确值0x3f800000。
你可能还会纳闷,float同样在内存中只占4个字节,为什么指定%f时不会出错?答案就是如果你指定printf输出参数为%f,那么printf在内存中读取8字节,而不是仅仅是低地址的4 Byte. 写一段程序测试一下便知:

int main()
{
    int    a=1, b=2, c=3;
    printf("%f, %d\n", a,b,c);
    return 0;
}


输出结果:
0.00000, 3
故可知,%f其实读取的是8 byte, 只不过a, b在转化成float型的时候,都因为值太小而被当作0.00000输出。

posted on   紫金树下  阅读(506)  评论(0编辑  收藏  举报
编辑推荐:
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
阅读排行:
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· AI技术革命,工作效率10个最佳AI工具
< 2013年5月 >
28 29 30 1 2 3 4
5 6 7 8 9 10 11
12 13 14 15 16 17 18
19 20 21 22 23 24 25
26 27 28 29 30 31 1
2 3 4 5 6 7 8

点击右上角即可分享
微信分享提示