」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 超越 JavaScript - 為什麼 + 在程式設計上不等於

超越 JavaScript - 為什麼 + 在程式設計上不等於

發佈於2024-11-01
瀏覽:469

當開發者第一次遇到這個看似令人費解的結果時,JavaScript 經常被嘲笑:

0.1   0.2 == 0.30000000000000004

關於 JavaScript 處理數字的迷因很普遍,常常導致許多人相信這種行為是該語言所獨有的。

Beyond JavaScript - Why     doesn

然而,這個怪癖不僅限於 JavaScript。這是大多數程式語言處理浮點運算方式的結果。

例如,以下是來自 JavaGo 的程式碼片段,它們產生類似的結果:

Beyond JavaScript - Why     doesn

Beyond JavaScript - Why     doesn

電腦本身只能儲存整數。他們不懂分數。 (他們會怎麼做?電腦進行算術運算的唯一方法是打開或關閉一些燈。燈可以打開或關閉。它不能「半」亮!)他們需要某種表示浮點數的方法。由於這種表示方法並不完全準確,因此 0.1 0.2 通常不等於 0.3。

所有分母由數係基數的質因數組成的分數都可以清晰地表達,而任何其他分數都會有重複的小數。例如,在以10 為基數的數字系統中,可以清楚地表示1/2、1/4、1/5、1/10 等分數,因為每種情況下的分母均由2 或5(10 的質因數)組成然而,像1/3、1/6、1/7 這樣的分數都有循環小數。

同樣,在二進位系統中,像 1/2、1/4、1/8 這樣的分數都可以清晰地表達,而所有其他分數都有循環小數。當您對這些循環小數執行算術運算時,您最終會得到剩餘的內容,當您將計算機的數字二進製表示形式轉換為人類可讀的以 10 為基數的表示形式時,這些剩餘內容會繼續存在。這就是導致大致正確結果的原因。

既然我們已經確定這個問題並非 JavaScript 所獨有,那麼讓我們探討一下浮點數是如何在幕後表示和處理的,以了解為什麼會出現這種行為。

為了了解浮點數在底層是如何表示和處理的,我們首先必須了解IEEE 754浮點標準。

IEEE 754 標準是廣泛使用的規範,用於在電腦系統中表示浮點數並對其執行算術運算。它的創建是為了確保在各種計算平台上使用浮點運算時的一致性。大多數程式語言和硬體實作(CPU、GPU 等)都遵守此標準。

這是數字在 IEEE 754 格式中的表示方式:

Beyond JavaScript - Why     doesn

這裡s是符號位(0表示正數,1表示負數),M是尾數(保存數字的位數)和E 是決定數字小數位數的指數。

您將無法找到任何可以在此格式中精確表示數字(如 0.1、0.2 或 0.3)的 M 和 E 整數值。我們只能選擇給出最接近結果的 M 和 E 值。

這裡有一個工具,您可以用來確定 IEEE 754 十進制數表示法:https://www.h-schmidt.net/FloatConverter/IEEE754.html

IEEE 754 0.25 表示法:

Beyond JavaScript - Why     doesn

IEEE 754 分別表示 0.1 與 0.2:

Beyond JavaScript - Why     doesn
Beyond JavaScript - Why     doesn

請注意,0.25時轉換誤差為0,而0.1和0.2則為非零誤差。

IEEE 754 定義了以下表示浮點數的格式:

  • 單精確度(32 位元):1 位元符號,8 位元指數,23 位元尾數

  • 雙精確度(64 位元):1 位元符號,11 位元指數,52 位元尾數

為了簡單起見,讓我們考慮使用 32 位元的單精度格式。

0.1的32位表示為:

0 01111011 10011001100110011001101

這裡第一位代表符號(0在本例中表示正數),接下來的8位(01111011)代表指數,最後23位(10011001100110011001101)代表尾數。

這不是準確的表示。它代表 ≈ 0.100000001490116119384765625

同樣,0.2的32位表示為:

0 01111100 10011001100110011001101

這也不是準確的表示。它代表 ≈ 0.20000000298023223876953125

加入後,結果是:

0 01111101 11001101010011001100110 

十進位表示中的 ≈ 0.30000001192092896

總之,看似令人困惑的結果 0.1 0.2 不產生 0.3 並不是 JavaScript 特有的異常現象,而是跨程式語言的浮點運算限制的結果。這種行為的根源在於數字的二進位表示形式,這在處理某些分數時本質上會導致精確度錯誤。

版本聲明 本文轉載於:https://dev.to/umangsinha12/beyond-javascript-why-01-02-doesnt-equal-03-in-programming-2bf3?1如有侵犯,請聯絡[email protected]刪除
最新教學 更多>
  • 如何使用 mysqli_pconnect() 在 PHP 中實作 MySQL 連線池?
    如何使用 mysqli_pconnect() 在 PHP 中實作 MySQL 連線池?
    MySQL 的 PHP 連線池在 PHP 中,維護資料庫連線會影響效能。為了優化這一點,開發人員經常考慮使用連接池技術。 MySQL 的連線池MySQL 沒有內建的連線池機制。然而,MySQLi 擴充功能提供了mysqli_pconnect() 函數,其作用與mysqli_connect() 類似,...
    程式設計 發佈於2024-11-07
  • 將 HTMX 加入 GO
    將 HTMX 加入 GO
    HTMX 是 intercooler.js 的後繼者,用於使用 HTTP 指令擴充 HTML,而無需編寫 API。現在,我知道一開始我說我要刪除抽象層,但是我更多的是系統/工具程式設計師,所以我仍然需要一些抽象,直到我掌握了底層實際發生的情況。 基本概念 HTMX 部署 AJAX ...
    程式設計 發佈於2024-11-07
  • 發現 itertools
    發現 itertools
    Itertools 是最有趣的 Python 函式庫之一。它包含一系列受函數式語言啟發的函數,用於與迭代器一起使用。 在這篇文章中,我將提到一些最引起我注意並且值得牢記的內容,以免每次都重新發明輪子。 數數 好幾次我都實現了無限數(好吧,結束了 明確地在某一點用中斷)使用 whi...
    程式設計 發佈於2024-11-07
  • 為什麼每個人都應該學習 Go(即使您認為生活中不需要另一種語言)
    為什麼每個人都應該學習 Go(即使您認為生活中不需要另一種語言)
    啊,Go,编程语言。您可能听说过,也许是从办公室里一位过于热情的开发人员那里听说过的,他总是不停地谈论他们的 API 现在有多“快得惊人”。当然,您已经涉足过其他语言,也许您会想:“我真的需要另一种语言吗?”剧透警报:是的,是的,你知道。 Go 就是那种语言。让我以最讽刺、最真诚的方式为你解释一下。...
    程式設計 發佈於2024-11-07
  • 如何計算 Pandas 中多列的最大值?
    如何計算 Pandas 中多列的最大值?
    在Pandas 中尋找多列的最大值假設您有一個包含多列的資料框,並且希望建立一個包含兩個或多個列中的最大值的新列現有的列。例如,給定A 列和B 列,您需要建立C 列,其中:C = max(A, B)要完成此任務:使用max 函數和axis=1 計算指定列中每行的最大值:df[["A&quo...
    程式設計 發佈於2024-11-07
  • 如何在 PHP 中從目錄中檢索檔案名稱?
    如何在 PHP 中從目錄中檢索檔案名稱?
    從 PHP 中的目錄中擷取檔案如何在 PHP 中存取目錄中的檔案名稱?事實證明,確定正確的命令具有挑戰性。這個問題旨在為尋求類似解決方案的個人提供幫助。 PHP提供了幾種從目錄獲取文件清單的方法:DirectoryIterator(建議)此類允許對目錄中的文件進行迭代:foreach (new Di...
    程式設計 發佈於2024-11-07
  • 使用 Linq、Criteria API 和 Query Over 擴充 NHibernate 的 Ardalis.Specification
    使用 Linq、Criteria API 和 Query Over 擴充 NHibernate 的 Ardalis.Specification
    Ardalis.Specification is a powerful library that enables the specification pattern for querying databases, primarily designed for Entity Framework Cor...
    程式設計 發佈於2024-11-07
  • PYTHON:OOP {初學者版}
    PYTHON:OOP {初學者版}
    Python:物件導向程式設計[OOP]:是一種程式設計範式(模型),使用物件和類別來建立軟體一種模擬現實世界實體和關係的方法。這是基於物件可以包含資料和操作該資料的程式碼的想法。 關於物件導向編程,您需要了解一些關鍵概念: 班級 目的 屬性 方法 遺產 封裝 多態性 抽象 下面的範例是一個幫助您...
    程式設計 發佈於2024-11-07
  • Neo.mjs:一個高效能開源 JavaScript 框架。
    Neo.mjs:一個高效能開源 JavaScript 框架。
    在瀏覽 GitHub 並尋找可協作的開源專案時,我發現了 Neo.mjs。我對這個計畫產生了興趣,並開始更多地研究這個新框架。我想在這篇文章中分享我發現的所有內容。 什麼是 Neo.mjs? Neo.mjs 旨在建立高效能、資料驅動的 Web 應用程序,重點在於利用 Web Wor...
    程式設計 發佈於2024-11-07
  • 將 Azure Functions 部署到 Azure 容器應用程式的兩種方法的比較
    將 Azure Functions 部署到 Azure 容器應用程式的兩種方法的比較
    昨天,我寫了一篇題為「在 Azure 容器應用程式上部署 Java Azure Function」的文章。 在那篇文章中,我提到使用 Azure 的整合管理功能,我想澄清這意味著什麼以及它與本文中先前的方法有何不同。 舊方法:使用 az containerapp create 創...
    程式設計 發佈於2024-11-07
  • 如何使用 MinGW 在 Windows 上建置 GLEW?逐步指南。
    如何使用 MinGW 在 Windows 上建置 GLEW?逐步指南。
    使用MinGW 在Windows 上建立GLEW:綜合指南使用GLEW,這是一個無縫整合OpenGL 和WGL 函數的純頭文件庫,使用MinGW 增強Windows 上OpenGL 應用程式的開發。為了使用 MinGW 有效建置 GLEW,需要一組特定的命令和步驟。 首先,建立兩個名為 lib 和 ...
    程式設計 發佈於2024-11-07
  • 如何使用 CSS 創建帶有對角線的雙色調背景?
    如何使用 CSS 創建帶有對角線的雙色調背景?
    使用對角線創建雙色調背景要使用CSS 實現由對角線分為兩部分的背景,請執行以下操作這些步驟:1。建立兩個 Div:建立兩個單獨的 div 來表示兩個背景部分。 2.設定 Div 樣式:將下列 CSS 套用至 div:.solid-div { background-color: [solid co...
    程式設計 發佈於2024-11-07
  • 文件的力量:閱讀如何改變我在 JamSphere 上使用 Redux 的體驗
    文件的力量:閱讀如何改變我在 JamSphere 上使用 Redux 的體驗
    作為開發人員,我們經常發現自己一頭扎進新的庫或框架,渴望將我們的想法變為現實。跳過文件並直接跳到編碼的誘惑很強烈——畢竟,這有多難呢?但正如我透過建立音樂管理平台 JamSphere 的經驗所了解到的那樣,跳過這一關鍵步驟可能會將順利的旅程變成充滿挑戰的艱苦戰鬥。 跳過文檔的魅力 ...
    程式設計 發佈於2024-11-07
  • 如何在PHP多子網域應用中精準控制Cookie域?
    如何在PHP多子網域應用中精準控制Cookie域?
    在PHP 中控制Cookie 域和子域在PHP 中控制Cookie 域和子域建立多子網域網站時,必須控制會話cookie 的網域確保每個子網域的正確會話管理。然而,手動設定網域時,PHP 的 cookie 處理似乎存在差異。 header("Set-Cookie: cookiename=c...
    程式設計 發佈於2024-11-07
  • 如何取得已安裝的 Go 軟體包的完整清單?
    如何取得已安裝的 Go 軟體包的完整清單?
    檢索Go 中已安裝軟體包的綜合清單在多台電腦上傳輸Go 軟體包安裝時,有必要取得詳細的清單所有已安裝的軟體包。本文概述了此任務的簡單且最新的解決方案。 解決方案:利用“go list”與過時的答案相反,當前的建議列出Go 中已安裝的軟體包是使用“go list”命令。透過指定三個文字句點 ('...
    程式設計 發佈於2024-11-07

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3