C语言printf输出浮点数的精度问题

本文深入探讨了浮点数在计算机中的表示精度问题,通过0.3在C语言中的float和double类型输出,揭示了二进制转换和IEEE754标准的影响。分析了不同精度下浮点数的四舍五入规则,并解释了为何输出结果有时看似精确,实则存在微小误差。
摘要由CSDN通过智能技术生成

相信学过C语言的人对printf这个函数再熟悉不过了,但是有些问题估计很多人不一定说得清楚。最近看《数值分析》,使得我对数值精度问题有了新的认识,还是那句话数学才是最直截了当的武器,没有数学依据只能停留在表面,难以知其所以然!

让我们看看下面几个案例:

float f = 0.3;
printf("%f",f);

没错这里可以输出:0.300000
这是C语言的默认输出6位,%f的输出让你觉得完全没问题,等等,似乎不太对,不是说计算机不能精确表示浮点数吗?这里怎么能精确表示呢?稍等,请看下面的代码段。

float f = 0.3;
printf("%.8f",f);

这里输出的是:0.30000001
终于露出了马脚,果然还是不精确啊,可是为什么是这个结果呢? 接着往下看

double f = 0.3;
printf("%.8lf",f);

这里输出的是:0.30000000
好奇怪,这里把float换成了double为什么又是正确的呢?输出更多的位看看呢?

double f = 0.3;
printf("%.18lf",f);

这里输出的是:0.299999999999999990
这下又回到了不精确的值了,这是为什么呢?

下面就来讲讲这个问题:

我们不妨先把0.3转换成二进制看看是什么样子的,推荐一个可以实现多种进制转换的网站:http://www.sojson.com/hexconvert.html

0.3的二进制结果为:0.0100110011001100110011001100110011001100110011001101
当然远不止这么多位,准确的讲应该是一个无限小数。根据 IEEE754标准 ,对于单精度浮点数来讲,尾数是23位,因此截取23位二进制,当然是从左到右第一个不为0的数字开始,

因此0.3的float 的尾数为:0.010011001100110011001101

注意看这里的最后一位是1,而不是0, 而原始二进制串中是0, 这里就采用了四舍五入的原则,回忆一下,在10进制中比如0.57四舍五入保留一位小数的结果是0.6,那就是因为第2位小数是7,超过了5所以进1位, 在二进制中就看下一位是1还是0,如果是1就进位。原始二进制串中的第24位为1,因此进位到23位,因此23位变为1.

那么我们接下来吧 0.3的float 的尾数即0.010011001100110011001101 转换为10进制看看。

结果是:0.300000011920928955078125

这就是为什么

printf("%f",f);     //可以得到 0.300000
printf("%.8f",f);   //可以得到 0.30000001

可以再多输出一些位数:

printf("%.15f",f);  //可以得到 0.300000011920929

其实可以看到,这里输出也有四舍五入将 0.300000011920928955078125 保留15位小数得到 0.300000011920929

至于第4个双精度输出为什么结果是小于0.3,那是因为它的尾数有52位,因此在截取二进制位的时候,因为53位是0,所以52位数字保持不变,实际上比原来的数要小一些,因为53位之后的数位依然有很多1存在。

那么还有一个疑问,为什么

double f = 0.3;
printf("%.8lf",f);

输出的是:0.30000000
而不是:0.29999999

这里是因为C语言在输出处理的时候,再次使用了四舍五入。这也是为什么我们用%f输出0.3能看到0.30000的原因。

  • 27
    点赞
  • 62
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
引用\[1\]中提到了C语言中使用printf输出浮点数问题。在C语言中,可以使用格式化字符串"%f"来输出浮点数。然而,由于计算机的浮点数表示存在精度问题,所以输出的结果可能不够精确。为了增加输出精度,可以使用指定精度的格式化字符串,例如"%.8f"表示输出浮点数并保留8位小数。这样可以提高输出精度。\[1\] 另外,引用\[3\]中提到了printf在传递参数时,若实参为float型数据,编译时会自动转换为double类型。因此,在使用printf输出浮点数时,可以将float型数据转换为double型数据,以提高输出精度。例如,可以使用"(double)10.9f"来将float型数据转换为double型数据并输出。\[3\] 综上所述,使用printf输出浮点数时,可以使用格式化字符串"%f"并指定精度,同时将float型数据转换为double型数据以提高输出精度。 #### 引用[.reference_title] - *1* [C语言printf输出浮点数精度问题](https://blog.csdn.net/u011426016/article/details/119868622)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down28v1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [printf输出浮点数遇到的问题 %f的问题](https://blog.csdn.net/lileiyuan1985/article/details/130862201)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down28v1,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [printf 输出浮点数](https://blog.csdn.net/weixin_30732825/article/details/94995182)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down28v1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Researcher-Du

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值