Python浮点型(float)运算结果不正确的解决方案

作者:诸子流 时间:2023-10-04 16:57:44 

一、问题说明

以前对浮点数运行没有没有太在意。昨天同事要求把百分比结果保存到文件上,然后就以保存1位小数的形式给他保存了。

但是今天同事运行时问能不能统一以一位小数的形式保存,当时觉得很奇怪昨天就是以一位小数形式存的怎么还会提这种要求呢。

其给回的截图确实是部分是一位小数的,但一部分是很长的。查看代码都统一如下格式:


# 使用round保留三位小数,然后乘以100,最后格式化为带百分号的字符串
rate=f"{round(x/y,3) * 100}%"

代码上没看出什么问题,直接运行确实是有些结果是一长串的。进行调试发现当x为37y为76时即会出现问题,如下图所示:

Python浮点型(float)运算结果不正确的解决方案

进行步骤拆分,发现round方法没有问题,问题在浮点数乘以100上(同时如下图可以看到也不是所有浮点数乘都有问题)

Python浮点型(float)运算结果不正确的解决方案

二、原因探究

搞不清原因,直到看到这篇文章:https://www.programiz.com/python-programming/numbers

大意是说二进制对很多浮点数无法准确表示只能用一个近似值代替,而当使用这些以近似值代替的浮点数进行进算时本质上是这些进似值参与了运算,出来的结果也就是进似值运算后的结果。

也就是说,一是这不是乘100的问题也不是乘法的问题而是整个浮点数运算都有问题,二是这不是python的问题是计算机浮点数存储的问题像C、Java等其他计算机语言进行运算都会有问题。

Python浮点型(float)运算结果不正确的解决方案

可能有人会疑惑:为什么二进制可以表示2不能表示0.2呢?

这是因为数值和字符串是不一样的,如果是字符串那么表示2.2点的左右两边的2编码是一样的就可以了(如ASCII码:504650),但数值不是这样,数值的整数部分和小数部分需要一个统一的表示形式,那就是加权位计数法。

整数部分都要以2的0次方(20)到2的无穷次方(2∞)表示,这没有问题,只要长度足够就能表示出所有奇数和偶数。2 = 1 * 21 + 0 * 20 = 10

小数部分都要以2的-1次方(2-1)到2的负无穷次方(2-∞)表示,这就有问题,因为比如2-1...2-∞不管怎么组合都不能完全等于0.2。0.2 = 0 * 2-1 + 0 * 2-2 + 1 * 2-3 ...

三、处理办法

这情况让我想起上份工作局方领导的一句话,应该是“可以理解但不能接受”。

原理上二进制无法精确表示一些浮点数可以理解,但是就这么返回个显然错误的结果给用户那是无法接受的。

python提供了Decimal()方法让浮点运算结果可以和人平时运算的结果一样。(Decimal本质应该还是通过加长长度提高精度)

Python浮点型(float)运算结果不正确的解决方案


# Decimal传字符串才能准确表示,所以需要先用str()把round()的结果转为字符串
rate=f"{Decimal(str(round(x/y,3))) * 100}%"

# 其实上边的结果出来是48.700%的形式,即三位小数的形式并不太符合我们保留一位小数的想法,真正符合想法得下面这样
# rate=f"{round(Decimal(str(round(x/y,3))) * 100, 1)}%"

# 其实我们说了这么多,我们都是建立在决定保留多少位再乘100这个前提下,倘若我们先乘100后决定保留几位那都不需要用Decimal
# rate=f"{round(x/y*100,1)}%"

Python浮点型(float)运算结果不正确的解决方案

参考:

https://www.programiz.com/python-programming/numbers

https://docs.python.org/3.7/library/decimal.html

来源:https://www.cnblogs.com/lsdb/p/10600815.html

标签:Python,浮点型,float,运算
0
投稿

猜你喜欢

  • 讲解Oracle数据库中结束死锁进程的一般方法

    2024-01-17 01:16:00
  • matplotlib绘制直方图的基本配置(万能模板案例)

    2021-04-21 02:35:42
  • vue父子组件的互相传值和调用

    2024-04-09 10:47:51
  • 浅析Python 中的 WSGI 接口和 WSGI 服务的运行

    2023-02-18 14:45:40
  • 如何将Yolov5的detect.py修改为可以直接调用的函数详解

    2021-12-12 22:21:28
  • 实用的PHP带公钥加密类分享(每次加密结果都不一样哦)

    2024-05-11 10:06:15
  • asp如何通过表单创建一个Word?

    2010-06-07 20:56:00
  • python3 scrapy框架的执行流程

    2022-04-12 08:53:35
  • Python包,__init__.py功能与用法分析

    2021-06-17 22:19:54
  • 对pyqt5中QTabWidget的相关操作详解

    2021-12-15 16:54:54
  • 数据库应用经验:如何简单安装MySQL数据库

    2009-01-04 12:58:00
  • 使用Python打造一款间谍程序的流程分析

    2021-11-21 08:12:32
  • php下实现伪 url 的超简单方法[转]

    2024-05-02 17:08:18
  • sqlserver清空service broker中的队列的语句分享

    2011-09-30 11:33:35
  • python3+PyQt5 创建多线程网络应用-TCP客户端和TCP服务器实例

    2021-01-14 10:20:29
  • Go语言递归函数的具体实现

    2023-08-05 02:35:32
  • pandas.read_csv参数详解(小结)

    2022-03-10 21:00:35
  • Golang中数据结构Queue的实现方法详解

    2024-04-26 17:20:00
  • 在SQL Server计算机上运行病毒扫描软件

    2009-01-19 14:38:00
  • 利用Java正则表达式校验邮箱与手机号

    2023-07-03 15:14:54
  • asp之家 网络编程 m.aspxhome.com