Python 浮點數的冷知識

来源:https://www.cnblogs.com/pythonista/archive/2019/09/21/11565135.html
-Advertisement-
Play Games

本周的 上分享了一篇小文章,它裡面提到的冷知識很有意思,我稍作補充,分享給大家。 它提到的部分問題,讀者們可以先思考下: 若兩個元組相等,即 a==b 且 a is b,那麼相同索引的元素(如 a[0] 、b[0])是否必然相等呢? 若兩個對象的 hash 結果相等,即 hash(a) == has ...


本周的PyCoder's Weekly 上分享了一篇小文章,它裡面提到的冷知識很有意思,我稍作補充,分享給大家。

它提到的部分問題,讀者們可以先思考下:

  • 若兩個元組相等,即 a==b 且 a is b,那麼相同索引的元素(如 a[0] 、b[0])是否必然相等呢?
  • 若兩個對象的 hash 結果相等,即 hash(a) == hash(b),那麼它們是否必然相等呢?

答案當然都為否(不然就不叫冷知識了),大家可以先嘗試回答一下,然後再往下看。

-----思考分割線-----

好了,先來看看第一個問題。兩個相同的元組 a、b,它們有如下的關係:

>>> a = (float('nan'),)
>>> b = a
>>> a   # (nan,)
>>> b   # (nan,)

>>> type(a), type(b)
(<type 'tuple'>, <type 'tuple'>)

>>> a == b
True

>>> a is b  # 即 id(a) == id(b)
True

>>> a[0] == b[0]
False

以上代碼表明:a 等於 b(類型、值與 id 都相等),但是它們的對位元素卻不相等。

兩個元組都只有一個元素(逗號後面沒有別的元素,這是單元素的元組的表示方法,即 len(a)==1 )。float() 是個內置函數,可以將入參構造成一個浮點數。

為什麼會這樣呢?先查閱一下文檔,這個內置函數的解析規則是:

sign           ::=  "+" | "-"
infinity       ::=  "Infinity" | "inf"
nan            ::=  "nan"
numeric_value  ::=  floatnumber | infinity | nan
numeric_string ::=  [sign] numeric_value

它在解析時,可以解析前後的空格、首碼的加減號(+/-)、浮點數,除此之外,還可以解析兩類字元串(不區分大小寫):"Infinity"或"inf",表示無窮大數;“nan”,表示不是數(not-a-number),確切地說,指的是除了數以外的所有東西。

前面分享的第一個冷知識就跟“nan”有關,作為整體,兩個元組相等,但是它們唯一的元素卻不相等。之所以會這樣,因為“nan”表示除了數以外的東西,它是一個範圍,所以不可比較。

作為對比,我們來看看兩個“無窮大的浮點數”是什麼結果:

>>> a = (float('inf'),)
>>> b = a
>>> a   # (inf,)
>>> b   # (inf,)

>>> a == b  # True
>>> a is b  # True
>>> a[0] == b[0]  # True

註意最後一次比較,它跟前面的兩個元組恰好相反,由此,我們可以得出結論:兩個無窮大的浮點數,數值相等,而兩個“不是數的東西”,數值不相等。

化簡一下,可以這樣看:

>>> a = float('inf')
>>> b = float('inf')
>>> c = float('nan')
>>> d = float('nan')

>>> a == b  # True
>>> c == d  # False

以上就是第一個冷知識的揭秘。接著看第二個:

>>> hash(float('nan')) == hash(float('nan'))
True

前面剛說了兩個“不是數的東西”不相等,這裡卻顯示它們的哈希結果相等,這挺違背常理的。

我們可以推理出一條簡單的結論:不相等的兩個對象,其哈希結果可能相等。

原因在於,hash(float('nan')) 的結果等於 0,它是個固定值,作比較時當然就相等了。

其實,關於 hash() 函數,還埋了一個彩蛋:

>>> hash(float('inf'))  # 314159
>>> hash(float('-inf')) # -314159

有沒有覺得這個數值很熟悉啊?它正是圓周率的前五位 3.14159,去除小數點後的結果。在早期的 Python 版本中,負無窮大數的哈希結果其實是 -271828,正是取自於自然對數 e。這兩個數都是硬編碼在 Python 解釋器中的,算是某種致敬吧。

由於 float('nan') 的哈希值相等,這通常意味著它們不可以作為字典的不同鍵值,但是事實卻出人意料:

>>> a = {float('nan'): 1, float('nan'): 2}
>>> a
{nan: 1, nan: 2}

# 作為對比:
>>> b = {float('inf'): 1, float('inf'): 2}
>>> b
{inf: 2}

如上所示,兩個 nan 鍵值在表示上一模一樣(註意,它們沒有用引號括起來),它們可以共存,而 inf 卻只能歸併成一個,再次展示出了 nan 的神奇。

好了,兩個很冷的小知識分享完畢,背後的原因都在於 float() 取浮點數時,Python 允許了 nan(不是數)的存在,它表示不確切的存在,所以導致了這些奇怪的結果。

最後,我們作下小結:

  • 包含 float('nan') 的兩個元組,當做整體作比較時,結果相等;兩個相等的元組,其對位的元素可能不相等
  • float('nan') 表示一個“不是數”的東西,它本身不是確定值,兩個對象作比較時不相等,但是其哈希結果是固定值,作比較時相等;可用作字典的鍵值,而且是不衝突的鍵值
  • float('inf') 表示一個無窮大的浮點數,可看作確定的值,兩個對象做比較時相等,其哈希結果也相等;可用作字典的鍵值,但是會產生衝突
  • float('nan') 的哈希結果為 0,float('inf') 的哈希結果為 314159

參考資料:

https://docs.python.org/3/library/functions.html#float

https://www.pythondoeswhat.com/2019/09/welcome-to-float-zone.html

公眾號【Python貓】, 本號連載優質的系列文章,有喵星哲學貓系列、Python進階系列、好書推薦系列、技術寫作、優質英文推薦與翻譯等等,歡迎關註哦。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 理解JVM記憶體分配策略 三大原則+擔保機制 JVM分配記憶體機制有三大原則和擔保機制 具體如下所示: 優先分配到eden區 大對象,直接進入到老年代 長期存活的對象分配到老年代 空間分配擔保 對象優先在Eden上分配 如何驗證對象優先在Eden上分配呢,我們進行如下實驗。 列印記憶體分配信息 首先代碼如 ...
  • 場景 Effective Java 中文版Java核心技術 捲Ⅰ 基礎知識(第8版)Java語言程式設計-進階篇(原書第8版)瘋狂Java講義Java從入門到精通 第三版Java編程思想第4版重構-改善既有代碼的設計Head First Java 中文高清版Java從入門到精通Java核心技術 捲Ⅱ ...
  • 2019-09-21-23:00:26 今天看了很多博客網的博客,看完覺得自己的博客真的是垃圾中的垃圾 新手不知道怎樣寫博客,我也很想寫好一篇能讓人看的博客,但是目前水平不夠 只能慢慢改,今天的博客還是按照自己的方式寫吧,明天開始學習怎麼寫一篇好的博客 但是感覺有點難,加油,但是自己寫博客也是為了記 ...
  • 一、發佈REST服務 二、使用RestTemplae調用服務 三、使用Feign調用服務 ...
  • 新聞 "宣告.NET Core 3.0第一個候選版本" ".NET Core 3.0第一個候選版本中ASP.NET Core與Blazor的更新" "F 的就業市場情形如何" "Finalization實現細節" "GitHub Pages編譯現在使用Checks API" "使用Visual St ...
  • 前言 越來越多的項目已經使用 "Java 8" 了,毫無疑問, "Java 8" 是Java自Java 5(發佈於2004年)之後的最重要的版本。這個版本包含語言、編譯器、庫、工具和 JVM 等方面的十多個新特性。在本文中我們將學習這些新特性,並用實際的例子說明在什麼場景下適合使用。 引用: 本文參 ...
  • 目的:寫給自己看的,不如其他大神全,以後慢慢修修改改 一:如何判斷數據是否垃圾數據 1.引用計數法 當創建一個對象時,為此對象分配一個引用計數器。當有其他對象引用這個對象時,計數器就+1。當引用失效了,計數器-1。 當一個對象的引用計數器=0時,此對象就可以被回收。 優點:原理簡單,實現方便。 缺點 ...
  • AJAX框架Day1 1、Ajax引言 a) 概念 Asynchronous Javascript And XML (非同步請求 javaScript And XML) b) 傳統的請求 2、非同步請求的開發 a) 非同步請求對象的創建 javascript: XmlHttpRequest對象 xhr C ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...