你怕是学了个假Python吧

571

之前就意识到一个问题,但是最近又有朋友提出来了,所以就想着干脆记录下来,分享给大家

啥问题呢?请看题:

也就是说,需要大家计算1.1-1的值,很多朋友会说:“emmm…这还不简单,玩我呢?不就是0.1嘛”

但是如果你用 python 去执行一下,会发现结果跟你想的不太一样,如下图:

这样大家是不是发现了什么问题?是的,浮点数在运算过程中并没有保证完全精确,是什么原因导致了这种现象呢?很多朋友就会窃喜:“这不就是 Python 的 bug 嘛~”

但实际上,这并不是 Python 中的 bug ,它和计算机硬件中如何处理浮点数有关。浮点数在计算机硬件中以二进制的形式存在,但是我们现在看到的都是十进制,而十进制的浮点数不能都完全精确的表示为二进制小数。

就比如说我们在十进制数中无法用小数精确表示 1/3 一样,在二进制数中也无法用小数精确表示 1/10。显然这样子的说明并没有十进制中的 1/3 那么直观,接下来我们尝试去计算一下二进制中的 1/10 :

十进制的整数位是二进制的整数位,十进制的小数位是二进制数的小数位。那现在我们拿到0.1

  • 整数部分为0

  • 小数部分为0.1,并顺序取值

    • 0.1*2=0.2<1取0

    • 0.2*2=0.4<1取0

    • 0.4*2=0.8<1取0

    • 0.8*2=1.6>1取1

    • 0.6*2=1.2>1取1

    • 0.2*2=0.4<1取0

有没有发现?在二进制下,1/10 是一个无限循环小数:0.00011001100110011…,显然这样的表示形式无法精确的表示浮点数,最终的结果是近似 1/10 。在使用 IEEE-754 浮点运算标准的计算机硬件上,Python 的浮点数映射为 IEEE-754 双精度浮点数,共包含 53 位精度(这里指的是二进制),在这个范围下,这个最接近 1/10 的结果是:3602879701896397/2∗∗55

这表示在计算机硬件中,1/10 的真实十进制数值为:0.1000000000000000055511151231257827021181583404541015625

那如何进行精确的浮点数运算呢?有朋友提出四舍五入可以解决。那我们来仔细看一下四舍五入真的可以解决这个问题吗?

四舍五入进行解决 在 python 中,使用 round(number[, ndigits]) 来进行四舍五入,其中 ndigits 表示保留几位小数,默认为0。

我们来看代码如下:

上面代码符合我们四舍五入的预期结果,但是不要着急,我们接着往下看:

这样看是不是有些问题,什么问题呢?按照四舍五入的话,round(1.15)会直接进为1.2,但是此时并没有,而是变为了1.1。这是为什么呢?

如果没有上面对浮点数的了解,仅从表象上很难去解释。我们已经知道了在计算机内部,对于一些浮点数是无法精确表示的,比如上面代码中 1.15,我们可以通过 format() 来看看它在计算机内部更加具体的数值: 看到这个结果,我们就恍然大悟,为什么看到的结果会是1.1了。

但是接下来,可能会更加的困惑,因为对于 0.5 来说,是完全可以直接转为二进制表示的。但是round(0.5)结果却为0?这是因为 round() 的工作原理为:对于 round(number[, ndigits]),如果 number 可以被正常处理,则它的值会被舍入到最接近的 10 的负 ndigits 次幂的倍数上,对于与两个倍数的差值(差值的绝对值)均相等的情况,则会选择两个倍数中的偶数。

这个规则,用我们熟悉的话来说即为“ 四舍六入五成双 ”。

使用decimal进行浮点数的精确计算 那我们在 Python 中怎么进行精确的浮点数计算呢,Python 标准库为我们提供了decimal 这个模块来解决这个问题,decimal 常用于需要精确处理浮点数的场合,比如银行账户金额、货币加减等。

In [17]: from decimal import Decimal

In [18]: 0.1-0.09 Out[18]: 0.010000000000000009

In [19]: Decimal('0.1')-Decimal('0.09') Out[19]: Decimal('0.01')

同样,我们可以使用它来查看对于不能精确表示的浮点数在计算机内部的具体数值:

这样就可以解决我们的困惑与问题