在知乎上上看到以下問題:html
1.該問題出現的緣由 ?java
2.爲什麼其餘編程語言,好比java中可能沒有js那麼明顯python
3.你們在項目中踩過浮點數精度的坑?git
4.最後採用哪些方案規避這個問題的?github
5.爲什麼採用改方案?chrome
alert(0.7+0.1); //輸出0.7999999999999999
以前本身答的不是滿意(對 陳嘉棟的回答 仍是滿意的),想對這個問題作個深刻淺出的總結npm
再看到這幾篇長文《[ JS 基礎 ] JS 浮點數四則運算精度丟失問題 (3)》、《JavaScript數字精度丟失問題總結》、《細說 JavaScript 七種數據類型》,略有所悟,整理以下:編程
這個問題並不僅是在Javascript中才會出現,任何使用二進制浮點數的編程語言都會有這個問題,只不過在 C++/C#/Java 這些語言中已經封裝好了方法來避免精度的問題,而 JavaScript 是一門弱類型的語言,從設計思想上就沒有對浮點數有個嚴格的數據類型,因此精度偏差的問題就顯得格外突出。segmentfault
JavaScript 中的數字類型只有 Number 一種,Number 類型採用 IEEE754 標準中的 「雙精度浮點數」 來表示一個數字,不區分整數和浮點數 (js位運算或許是爲了提高B格)。oracle
幾乎全部的編程語言浮點數都是都採用IEEE浮點數算術標準。java float 32 浮點數: 1bit符號 8bit指數部分 23bit尾數。推薦閱讀《JAVA 浮點數的範圍和精度》
IEEE-745浮點數表示法是一種能夠精確地表示分數的二進制示法,好比1/2,1/8,1/1024
十進制整數換成二進制通常都會:1=>1 2=>10 3=>101 4=>100 5=>101 6=>110
6/2=3…0
3/2=1…1
1/2=0…1
倒過來就是110
0.25的二進制
0.25*2=0.5 取整是0
0.5*2=1.0 取整是1
即0.25的二進制爲 0.01 ( 第一次所獲得爲最高位,最後一次獲得爲最低位)
0.8125的二進制
0.8125*2=1.625 取整是1
0.625*2=1.25 取整是1
0.25*2=0.5 取整是0
0.5*2=1.0 取整是1
即0.8125的二進制是0.1101(第一次所獲得爲最高位,最後一次獲得爲最低位)
0.1的二進制
0.1*2=0.2======取出整數部分0
0.2*2=0.4======取出整數部分0
0.4*2=0.8======取出整數部分0
0.8*2=1.6======取出整數部分1
0.6*2=1.2======取出整數部分1
0.2*2=0.4======取出整數部分0
0.4*2=0.8======取出整數部分0
0.8*2=1.6======取出整數部分1
0.6*2=1.2======取出整數部分1
接下來會無限循環
0.2*2=0.4======取出整數部分0
0.4*2=0.8======取出整數部分0
0.8*2=1.6======取出整數部分1
0.6*2=1.2======取出整數部分1
因此0.1轉化成二進制是:0.0001 1001 1001 1001…(無限循環)
0.1 => 0.0001 1001 1001 1001…(無限循環)
同理0.2的二進制是0.0011 0011 0011 0011…(無限循環)
在 IEEE754 中,雙精度浮點數採用 64 位存儲,即 8 個字節表示一個浮點數 。其存儲結構以下圖所示:
指數位能夠經過下面的方法轉換爲使用的指數值:
從存儲結構中能夠看出, 指數部分的長度是11個二進制,即指數部分能表示的最大值是 2047(2^11-1)
取中間值進行偏移,用來表示負指數,也就是說指數的範圍是 [-1023,1024] 。
所以,這種存儲結構可以表示的數值範圍爲 2^1024 到 2^-1023 ,超出這個範圍的數沒法表示 。2^1024 和 2^-1023 轉換爲科學計數法以下所示:
1.7976931348623157 × 10^308
5 × 10^-324
所以,JavaScript 中能表示的最大值是 1.7976931348623157 × 10308,最小值爲 5 × 10-324 。java雙精度類型 double也是如此。
這兩個邊界值能夠分別經過訪問 Number 對象的 MAX_VALUE 屬性和 MIN_VALUE 屬性來獲取:
Number.MAX_VALUE; // 1.7976931348623157e+308 Number.MIN_VALUE; // 5e-324
若是數字超過最大值或最小值,JavaScript 將返回一個不正確的值,這稱爲 「正向溢出(overflow)」 或 「負向溢出(underflow)」 。
Number.MAX_VALUE+1 == Number.MAX_VALUE; //true Number.MAX_VALUE+1e292; //Infinity Number.MIN_VALUE + 1; //1 Number.MIN_VALUE - 3e-324; //0 Number.MIN_VALUE - 2e-324; //5e-324
在 64 位的二進制中,符號位決定了一個數的正負,指數部分決定了數值的大小,小數部分決定了數值的精度。
IEEE754 規定,有效數字第一位默認老是1 。所以,在表示精度的位數前面,還存在一個 「隱藏位」 ,固定爲 1 ,但它不保存在 64 位浮點數之中。也就是說,有效數字老是 1.xx...xx 的形式,其中 xx..xx 的部分保存在 64 位浮點數之中,最長爲52位 。因此,JavaScript 提供的有效數字最長爲 53 個二進制位,其內部實際的表現形式爲:
(-1)^符號位 * 1.xx...xx * 2^指數位
這意味着,JavaScript 能表示並進行精確算術運算的整數範圍爲:[-2^53-1,2^53-1],即從最小值 -9007199254740991 到最大值 9007199254740991 之間的範圍 。
Math.pow(2, 53)-1 ; // 9007199254740991 -Math.pow(2, 53)-1 ; // -9007199254740991
能夠經過 Number.MAX_SAFE_INTEGER 和 Number.MIN_SAFE_INTEGER 來分別獲取這個最大值和最小值。
console.log(Number.MAX_SAFE_INTEGER) ; // 9007199254740991 console.log(Number.MIN_SAFE_INTEGER) ; // -9007199254740991
對於超過這個範圍的整數,JavaScript 依舊能夠進行運算,但卻不保證運算結果的精度。
Math.pow(2, 53) ; // 9007199254740992 Math.pow(2, 53) + 1; // 9007199254740992 9007199254740993; //9007199254740992 90071992547409921; //90071992547409920 0.923456789012345678;//0.9234567890123456
計算機中的數字都是以二進制存儲的,二進制浮點數表示法並不能精確的表示相似0.1這樣 的簡單的數字
若是要計算 0.1 + 0.2 的結果,計算機會先把 0.1 和 0.2 分別轉化成二進制,而後相加,最後再把相加獲得的結果轉爲十進制
但有一些浮點數在轉化爲二進制時,會出現無限循環 。好比, 十進制的 0.1 轉化爲二進制,會獲得以下結果:
0.1 => 0.0001 1001 1001 1001…(無限循環)
0.2 => 0.0011 0011 0011 0011…(無限循環)
而存儲結構中的尾數部分最多隻能表示 53 位。爲了能表示 0.1,只能模仿十進制進行四捨五入了,但二進制只有 0 和 1 , 因而變爲 0 舍 1 入 。 所以,0.1 在計算機裏的二進制表示形式以下:
0.1 => 0.0001 1001 1001 1001 1001 1001 1001 1001 1001 1001 1001 1001 1001 101
0.2 => 0.0011 0011 0011 0011 0011 0011 0011 0011 0011 0011 0011 0011 0011 001
用標準計數法表示以下:
0.1 => (−1)0 × 2^4 × (1.1001100110011001100110011001100110011001100110011010)2
0.2 => (−1)0 × 2^3 × (1.1001100110011001100110011001100110011001100110011010)2
在計算浮點數相加時,須要先進行 「對位」,將較小的指數化爲較大的指數,並將小數部分相應右移:
最終,「0.1 + 0.2」 在計算機裏的計算過程以下:
通過上面的計算過程,0.1 + 0.2 獲得的結果也能夠表示爲:
(−1)0 × 2−2 × (1.0011001100110011001100110011001100110011001100110100)2=>.0.30000000000000004
經過 JS 將這個二進制結果轉化爲十進制表示:
(-1)**0 * 2**-2 * (0b10011001100110011001100110011001100110011001100110100 * 2**-52); //0.30000000000000004
console.log(0.1 + 0.2) ; // 0.30000000000000004
這是一個典型的精度丟失案例,從上面的計算過程能夠看出,0.1 和 0.2 在轉換爲二進制時就發生了一次精度丟失,而對於計算後的二進制又有一次精度丟失 。所以,獲得的結果是不許確的。
浮點數丟失解決方案
咱們經常使用的分數(特別是在金融的計算方面)都是十進制分數1/10,1/100等。或許之後電路設計或許會支持十進制數字類型以免這些舍入問題。在這以前,你更願意使用大整數進行重要的金融計算,例如,要使用整數‘分’而不是使用小數‘元’進行貨比單位的運算
即在運算前咱們把參加運算的數先升級(10的X的次方)到整數,等運算完後再降級(0.1的X的次方)。
在java裏面有BigDecimal庫,js裏面有big.js js-big-decimal.js。固然BCD編碼就是爲了十進制高精度運算量制。
BCD編碼(通常指8421BCD碼形式)亦稱二進碼十進數或二-十進制代碼。用4位二進制數來表示1位十進制數中的0~9這10個數。通常用於高精度計算。好比會計制度常常須要對很長的數字串做準確的計算。相對於通常的浮點式記數法,採用BCD碼,既可保存數值的精確度,又可免去使電腦做浮點運算時所耗費的時間。
二進制在電路設計中物理上更易實現,由於電子器件大多具備兩種穩定狀態,好比晶體管的導通和截止,電壓的高和低,磁性的有和無等。而找到一個具備十個穩定狀態的電子器件是很困難的。
二進制規則簡單,十進制有55種求和與求積的運算規則,二進制僅有各有3種,這樣能夠簡化運算器等物理器件的設計。另外,計算機的部件狀態少,能夠加強整個系統的穩定性。
與邏輯量相吻合。二進制數0和1正好與邏輯量「真」和「假」相對應,所以用二進制數表示二值邏輯顯得十分天然。
可靠性高。二進制中只使用0和1兩個數字,傳輸和處理時不易出錯,於是能夠保障計算機具備很高的可靠性
我以爲主要仍是由於第一條。若是好比可以設計出十進制的元器件,那麼對於設計其運算器也再也不話下。
兩個簡單的浮點數相加
0.1 + 0.2 != 0.3 // true
toFixed 不會四捨五入(Chrome)
1.335.toFixed(2) // 1.33
再問問一個問題 :在js數字類型中浮點數的最高精度多少位小數?(16位 or 17位?……why?
JavaScript 能表示並進行精確算術運算的整數範圍爲:[-2^53-1,2^53-1],即從最小值 -9007199254740991 到最大值 9007199254740991 之間的範圍。'9007199254740991'.length//16
IEEE754 規定,有效數字第一位默認老是1 。所以,在表示精度的位數前面,還存在一個 「隱藏位」 ,固定爲 1 ,但它不保存在 64 位浮點數之中。也就是說,有效數字老是 1.xx...xx 的形式,其中 xx..xx 的部分保存在 64 位浮點數之中,最長爲52位 。因此,JavaScript 提供的有效數字最長爲 53 個二進制位
let a=1/3
a.toString();//"0.3333333333333333"
a.toString();.length//18
a*3===0.3333333333333333*3===1
0.3333333333333332*3!==1