JavaScript的高精度計算與JSON.parse的BIGINT

relsoul發表於2018-05-10

在JavaScript處理整數的時候會遇到某些特別奇怪的問題,比如後臺給你返回了一個超長的數字,然後js在計算的時候突然發現計算不對,不是後面為0就是計算得不到想要的結果.這裡涉及到一個很簡單的知識 也就是NUMBER的安全整數.

Number安全整數

Number.MAX_SAFE_INTEGER // 9007199254740991
9007199254740991+2 // 9007199254740992

複製程式碼

首先我們拿到了安全範圍內的數字 然後對他進行簡單的加法,正常的結果應該是9007199254740993 但是這裡的尾數只有2, 大家可以去測試一下也是挺有意思的。因為js中所有的整數都是用浮點型別(double-precision 64-bit binary format IEEE 754 value)導致的,雖然可以承受的範圍比較大,但是計算精度卻不怎麼好

計算精度

說完了安全整數這裡來簡單說說計算精度

0.1+0.2 //0.30000000000000004
複製程式碼

原理也是同上 那麼解決辦法也很簡單

使用math.js庫即可解決

// prevent round-off errors showing up in output
var ans = math.add(0.1, 0.2);       //  0.30000000000000004
math.format(ans, {precision: 14});  // '0.3'
複製程式碼

JSON.parse中遇到的BIGINT

比較常見的場景為後臺返回了一串JSON但是數字為BIGINT導致解析錯誤,一般為16位

var c='{"num": 90071992547409999}'
JSON.parse(c) // {num: 90071992547410000}
複製程式碼

比如我們使用jQuery的ajax開啟了json:true功能 jQuery會自動parse 這樣的話會造成精度不夠的問題,所以這裡可以使用json-bigint

var JSONbig = require('json-bigint');

var json = '{ "value" : 9223372036854775807, "v2": 123 }';
console.log('Input:', json);
console.log('');

console.log('node.js bult-in JSON:')
var r = JSON.parse(json);
console.log('JSON.parse(input).value : ', r.value.toString());
console.log('JSON.stringify(JSON.parse(input)):', JSON.stringify(r));

console.log('\n\nbig number JSON:');
var r1 = JSONbig.parse(json);
console.log('JSON.parse(input).value : ', r1.value.toString());
console.log('JSON.stringify(JSON.parse(input)):', JSONbig.stringify(r1));

// output

Input: { "value" : 9223372036854775807, "v2": 123 }

node.js bult-in JSON:
JSON.parse(input).value :  9223372036854776000
JSON.stringify(JSON.parse(input)): {"value":9223372036854776000,"v2":123}


big number JSON:
JSON.parse(input).value :  9223372036854775807
JSON.stringify(JSON.parse(input)): {"value":9223372036854775807,"v2":123}
複製程式碼

如果不想要引入額外的庫則可以參考這段函式

let stringedJSON = origJSON.replace(/:\s*([-+Ee0-9.]+)/g, ': "uniqueprefix$1"');

let o = JSON.parse(stringedJSON, (key, value) => {
  // only changing strings
  if (typeof value !== 'string') return value;
  // only changing number strings
  if (!value.startsWith('uniqueprefix')) return value;
  // chop off the prefix
  value = value.slice('uniqueprefix'.length);
  // pick your favorite arbitrary-precision library
  return new Big(value);
});
複製程式碼

參考文獻

相關文章