本文由雲+社區發表
相信你們在日常的 JavaScript 開發中,都有遇到過浮點數運算精度偏差的問題,好比 console.log(0.1+0.2===0.3)// false
。在 JavaScript 中,全部的數字包括整數和小數都是用 Number
類型來表示的。本文經過介紹 Number
的二進制存儲標準來理解浮點數運算精度問題,和理解 Number
對象的 MAX_VALUE
等屬性值是如何取值的,最後介紹了一些經常使用的浮點數精度運算解決方案。javascript
JavaScript Number 採用的是 IEEE 754 定義的 64 位雙精度浮點型來表示。具體的字節分配能夠先看一下引自維基百科的圖:java
從上圖中能夠看到,從高到低,64位被分紅3段,分別是:安全
指數位有 11 位,取值範圍是 0 到 2047。當指數位 e=0 或者 e=2017 時,根據有效數字位 f 是否爲 0 ,具備不一樣的特殊含義,具體見下表:spa
對於經常使用的 normal number, 爲了方便表示指數爲負數的狀況,因此,指數位數值大小作了一個 -1023 的偏移量。對於一個非 0 數字而言,,它的二進制的科學計數法裏的第一位有效數字固定是 1。這樣,一個雙精度浮點型數字的值就是code
對於 subnormal number,它能夠用來表示更加接近於 0 的數,它特殊的地方是有效數字位的前面補充的是 0 而不是 1,且指數爲偏移量是 -1022,因此值是:orm
知道了 Number 是如何存儲以後,Number 對象的屬性是如何取值的就明朗了。對象
Number.MAX_VALUE:可表示的最大的數,顯然 e 和 f 都取最大時能表示的數最大,值爲blog
Number.MIN_VALUE:可表示的最小的正數,用最小的 subnormal number 來表示。當 e = 0 ,f 的最後一位爲 1,其餘爲 0 時最小,值爲ip
Number.EPSILON : 表示 1 與 Number 可表示的大於 1 的最小的浮點數之間的差值。值爲ci
Number.MAXSAFEINTEGER:表示在 JavaScript 中最大的安全整數。能夠連續且精確被表示出來的整數成爲安全整數,好比 2^54 就不是個安全整數,由於它和 2^54+1 兩個數的表示是徹底同樣的,e=1077,f=0。 Math.pow(2,54)===Math.pow(2,54)+1// true
。整數轉化爲二進制後,小數點後是不會有數字的,而用二進制的科學計數法表示時,小數點後最多保留 52 位,加上前置的一個 1,有 53 位數字,因此當一個數轉化二進制時,若是位數超過 53 位,必然會截斷末尾的部分,即致使不能精確表示,即爲不安全整數。因此最小的會被截斷的整數是 100...001=2^53+1(中間有52個0)。這個數設爲 X,則比 X 小的整數都能被精確表示出來,再加上「連續」這個條件,因此 X-1 不是咱們要的答案,X-2 纔是。 Number.MAX_SAFE_INTEGER
最終值爲
Number.MINSAFEINTEGER:表示在 JavaScript 中最小的安全整數,對 Number.MAX_SAFE_INTEGER
取負值便可,值爲 -9007199254740991
如今看看 console.log(0.1+0.2===0.3)// false
這個問題,數字 0.1 轉化成二進制是 0.0001100110011... 即 1.10011001...1001 2^-4 (小數部分有52位,即有13個1001循環)。因爲第 53 位是 1,相似 10 進制的四捨五入,二進制是「零舍一入」,因此 0.1 的最終二進制科學計數法表示是 1.10011001...1010 2^-4,即二進制數值大小其實是 0.000110011001...10011010。下面的代碼驗證了這個值(打印出來的值,把最末尾的0去掉了):
var a = 0.1;console.log(a.toString(2)); //0.0001100110011001100110011001100110011001100110011001101
同理十進制數字 0.2 轉化爲二進制的最終值是 1.10011001...1010 2^-3 即 0.00110011...100111010;十進制 0.3 轉化位二進制的最終值是 1.00110011...0011 2^-2
var b = 0.2;console.log(b.toString(2)); //0.001100110011001100110011001100110011001100110011001101var c = 0.3;console.log(c.toString(2)); //0.010011001100110011001100110011001100110011001100110011
因此,0.1+0.2 的值即爲上面 0.1 和 0.2 對應的二進制數值的相加,以下圖所示
上圖中,對所得的和,「零舍一入」保留 52 位有效小數就是最終的值:0.01001100...110100(第 53 位是 1 ,因此往前進了 1),以下代碼所示。這個值與上文中的 0.3 的最終二進制表示的值明顯不相同,即解釋了 0.1 + 0.2 不等於 0.3 的根本緣由所在(實際上,這個值轉化爲 10 進制約等於 0.30000000000000004)。注:打印出來的長度是 54,由於有 52 位有效小數,前面是'0.01',長度是 4,最後去掉末尾的 2 個 0,因此最後打印出來的長度是 52+4-2 = 54。
var d = 0.1 + 0.2;console.log(d.toString(2)); //0.0100110011001100110011001100110011001100110011001101console.log(d.toString(2).length); // 54
關於 js 浮點數運算精度丟失的問題,不一樣場景能夠有不一樣的解決方案。 一、若是隻是用來展現一個浮點數的結果,則能夠借用 Number 對象的 toFixed 和 parseFloat 方法。下面代碼片斷中,fixed 參數表示要保留幾位小數,能夠根據實際場景調整精度。
function formatNum(num, fixed = 10) { return parseFloat(a.toFixed(fixed))}var a = 0.1 + 0.2;console.log(formatNum(a)); //0.3
二、若是須要進行浮點數的加減乘除等運算,由上文可知,在小於 Number.MAXSAFEINTEGER 範圍的整數是能夠被精確表示出來的,因此能夠先把小數轉化爲整數,運算獲得結果後再轉化爲對應的小數。好比兩個浮點數的加法:
function add(num1, num2) { var decimalLen1 = (num1.toString().split('.')[1] || '').length; //第一個參數的小數個數 var decimalLen2 = (num2.toString().split('.')[1] || '').length; //第二個參數的小數個數 var baseNum = Math.pow(10, Math.max(decimalLen1, decimalLen2)); return (num1 * baseNum + num2 * baseNum) / baseNum;}console.log(add(0.1 , 0.2)); //0.3
參考資料
此文已由做者受權騰訊雲+社區發佈