婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > 聊聊Python中的浮點數(shù)運算不準確問題

聊聊Python中的浮點數(shù)運算不準確問題

熱門標簽:長春人工智能電銷機器人官網(wǎng) 西藏快速地圖標注地點 地圖標注推廣單頁 如何在地圖標注文字 廈門crm外呼系統(tǒng)如何 女王谷地圖標注 ai地圖標注 百應ai電銷機器人鄭州 n400電話申請多少錢

大家好,老 Amy 來了。之前就意識到一個問題,但是最近又有朋友提出來了,所以就想著干脆記錄下來,分享給大家叭~

啥問題呢?請看題:

也就是說,需要大家計算1.1-1的值,很多朋友會說:“emmm…這還不簡單,玩我呢?不就是0.1嘛”

但是如果你用 python 去執(zhí)行一下,會發(fā)現(xiàn)結果跟你想的不太一樣,如下圖:

這樣大家是不是發(fā)現(xiàn)了什么問題?是的,浮點數(shù)在運算過程中并沒有保證完全精確,是什么原因導致了這種現(xiàn)象呢?很多朋友就會竊喜:“這不就是 Python 的 bug 嘛~”

但實際上,這并不是 Python 中的 bug ,它和計算機硬件中如何處理浮點數(shù)有關。浮點數(shù)在計算機硬件中以二進制的形式存在,但是我們現(xiàn)在看到的都是十進制,而十進制的浮點數(shù)不能都完全精確的表示為二進制小數(shù)。

就比如說我們在十進制數(shù)中無法用小數(shù)精確表示 1/3 一樣,在二進制數(shù)中也無法用小數(shù)精確表示 1/10。顯然這樣子的說明并沒有十進制中的 1/3 那么直觀,接下來我們嘗試去計算一下二進制中的 1/10 :

十進制的整數(shù)位是二進制的整數(shù)位,十進制的小數(shù)位是二進制數(shù)的小數(shù)位。那現(xiàn)在我們拿到0.1

整數(shù)部分為0

小數(shù)部分為0.1,并順序取值

0.1*2=0.21取0
0.2*2=0.41取0
0.4*2=0.81取0
0.8*2=1.6>1取1
0.6*2=1.2>1取1
0.2*2=0.41取0
…

有沒有發(fā)現(xiàn)?在二進制下,1/10 是一個無限循環(huán)小數(shù):0.00011001100110011…,顯然這樣的表示形式無法精確的表示浮點數(shù),最終的結果是近似 1/10 。在使用 IEEE-754 浮點運算標準的計算機硬件上,Python 的浮點數(shù)映射為 IEEE-754 雙精度浮點數(shù),共包含 53 位精度(這里指的是二進制),在這個范圍下,這個最接近 1/10 的結果是:

3602879701896397/2∗∗55

這表示在計算機硬件中,1/10 的真實十進制數(shù)值為:

0.1000000000000000055511151231257827021181583404541015625

那如何進行精確的浮點數(shù)運算呢?有朋友提出四舍五入可以解決。那我們來仔細看一下四舍五入真的可以解決這個問題嗎?

四舍五入進行解決

在 python 中,使用 round(number[, ndigits]) 來進行四舍五入,其中 ndigits 表示保留幾位小數(shù),默認為0。

我們來看代碼如下:

In [10]: round(0.6)
Out[10]: 1
In [11]: round(0.65,1)
Out[11]: 0.7
In [12]: round(0.64,1)
Out[12]: 0.6

上面代碼符合我們四舍五入的預期結果,但是不要著急,我們接著往下看:

In [13]: round(1.15,1)
Out[13]: 1.1
In [14]: round(0.5)
Out[14]: 0
In [15]: round(1.5)
Out[15]: 2

這樣看是不是有些問題,什么問題呢?按照四舍五入的話,round(1.15)會直接進為1.2,但是此時并沒有,而是變?yōu)榱?.1。這是為什么呢?

如果沒有上面對浮點數(shù)的了解,僅從表象上很難去解釋。我們已經(jīng)知道了在計算機內部,對于一些浮點數(shù)是無法精確表示的,比如上面代碼中 1.15,我們可以通過 format() 來看看它在計算機內部更加具體的數(shù)值:

In [16]: format(1.15,".51f")
Out[16]: '1.149999999999999911182158029987476766109466552734375'

看到這個結果,我們就恍然大悟,為什么看到的結果會是1.1了。

但是接下來,可能會更加的困惑,因為對于 0.5 來說,是完全可以直接轉為二進制表示的。但是round(0.5)結果卻為0?這是因為 round() 的工作原理為:對于 round(number[, ndigits]),如果 number 可以被正常處理,則它的值會被舍入到最接近的 10 的負 ndigits 次冪的倍數(shù)上,對于與兩個倍數(shù)的差值(差值的絕對值)均相等的情況,則會選擇兩個倍數(shù)中的偶數(shù)。

# 最接近的10的負0次冪的倍數(shù)為0、1,并與0、1差值的絕對值相同,選擇偶數(shù)0
>>> round(0.5) 
0
# 最接近的10的負2次冪的倍數(shù)為0.12、0.13,并與0.12、0.13的差值的絕對值相同,選擇偶數(shù)0.12
>>> round(0.125, 2) 
0.12
# 最接近的10的負2次冪的倍數(shù)為0.13
>>> round(0.12548828125, 2) 
0.13

這個規(guī)則,用我們熟悉的話來說即為“ 四舍六入五成雙 ”。

使用decimal進行浮點數(shù)的精確計算

那我們在 Python 中怎么進行精確的浮點數(shù)計算呢,Python 標準庫為我們提供了decimal 這個模塊來解決這個問題,decimal 常用于需要精確處理浮點數(shù)的場合,比如銀行賬戶金額、貨幣加減等。

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')

同樣,我們可以使用它來查看對于不能精確表示的浮點數(shù)在計算機內部的具體數(shù)值:

In [20]: Decimal.from_float(1.1)
Out[20]: Decimal('1.100000000000000088817841970012523233890533447265625')
In [21]: Decimal.from_float(0.1)
Out[21]: Decimal('0.1000000000000000055511151231257827021181583404541015625')

這樣就可以解決我們的困惑與問題啦~

補充:python做浮點數(shù)運算時的坑記錄

很顯然,這個計算結果是不對的,而且偏離實際值十分遠。。。。。。。。

太坑人了這。

本來想自動截取計算得到的圖片尺寸,但是這計算結果,坑害了半天的查找錯誤過程!!!!

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。

您可能感興趣的文章:
  • Python如何執(zhí)行精確的浮點數(shù)運算
  • python中實現(xiàn)精確的浮點數(shù)運算詳解
  • Python中的浮點數(shù)原理與運算分析
  • Python雙精度浮點數(shù)運算并分行顯示操作示例

標簽:廊坊 內江 拉薩 綿陽 亳州 黔東 渭南 興安盟

巨人網(wǎng)絡通訊聲明:本文標題《聊聊Python中的浮點數(shù)運算不準確問題》,本文關鍵詞  聊聊,Python,中的,浮,點數(shù),;如發(fā)現(xiàn)本文內容存在版權問題,煩請?zhí)峁┫嚓P信息告之我們,我們將及時溝通與處理。本站內容系統(tǒng)采集于網(wǎng)絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《聊聊Python中的浮點數(shù)運算不準確問題》相關的同類信息!
  • 本頁收集關于聊聊Python中的浮點數(shù)運算不準確問題的相關信息資訊供網(wǎng)民參考!
  • 推薦文章
    主站蜘蛛池模板: 汝城县| 桦南县| 和龙市| 库尔勒市| 平度市| 宁化县| 东平县| 娱乐| 远安县| 宁城县| 普定县| 广汉市| 安泽县| 荔浦县| 上虞市| 南汇区| 卓资县| 太仆寺旗| 莱西市| 泰安市| 盘山县| 砀山县| 永嘉县| 涿鹿县| 潞西市| 卢龙县| 泰安市| 金湖县| 普兰店市| 漳浦县| 手机| 新乡县| 枣阳市| 二连浩特市| 清水河县| 宝丰县| 韩城市| 武隆县| 济阳县| 沈阳市| 江川县|