原文:www.smashingmagazine.com/2019/07/ess…javascript
譯者:前端小智html
阿里雲最近在作活動,低至2折,有興趣能夠看看:promotion.aliyun.com/ntms/yunpar…前端
爲了保證的可讀性,本文采用意譯而非直譯。java
BigInt
數據類型的目的是比Number
數據類型支持的範圍更大的整數值。在對大整數執行數學運算時,以任意精度表示整數的能力尤其重要。使用BigInt
,整數溢出將再也不是問題。git
此外,能夠安全地使用更加準確時間戳,大整數ID等,而無需使用變通方法。 BigInt
目前是第3階段提案, 一旦添加到規範中,它就是JS 第二個數字數據類型,也將是 JS 第8種基本數據類型:程序員
在本文中,我們將詳細介紹BigInt
,看看它如何解決使用Number
類型的限制。github
對於學過其餘語言的程序員來講,JS中缺乏顯式整數類型經常使人困惑。許多編程語言支持多種數字類型,如浮點型、雙精度型、整數型和雙精度型,但JS卻不是這樣。在JS中,按照IEEE 754-2008標準的定義,全部數字都以雙精度64位浮點格式表示。編程
在此標準下,沒法精確表示的很是大的整數將自動四捨五入。確切地說,JS 中的Number
類型只能安全地表示-9007199254740991 (-(2^53-1))
和9007199254740991(2^53-1)
之間的整數,任何超出此範圍的整數值均可能失去精度。api
console.log(9999999999999999); // → 10000000000000000
複製代碼
該整數大於JS Number 類型所能表示的最大整數,所以,它被四捨五入的。意外四捨五入會損害程序的可靠性和安全性。這是另外一個例子:數組
// 注意最後一位的數字
9007199254740992 === 9007199254740993; // → true
複製代碼
JS 提供Number.MAX_SAFE_INTEGER
常量來表示 最大安全整數,Number.MIN_SAFE_INTEGER
常量表示最小安全整數:
const minInt = Number.MIN_SAFE_INTEGER;
console.log(minInt); // → -9007199254740991
console.log(minInt - 5); // → -9007199254740996
// notice how this outputs the same value as above
console.log(minInt - 4); // → -9007199254740996
複製代碼
爲了解決這些限制,一些JS開發人員使用字符串類型表示大整數。 例如,Twitter API 在使用 JSON 進行響應時會向對象添加字符串版本的 ID。 此外,還開發了許多庫,例如 bignumber.js,以便更容易地處理大整數。
使用BigInt,應用程序再也不須要變通方法或庫來安全地表示Number.MAX_SAFE_INTEGER
和Number.Min_SAFE_INTEGER
以外的整數。 如今能夠在標準JS中執行對大整數的算術運算,而不會有精度損失的風險。
要建立BigInt
,只需在整數的末尾追加n便可。比較:
console.log(9007199254740995n); // → 9007199254740995n
console.log(9007199254740995); // → 9007199254740996
複製代碼
或者,能夠調用BigInt()
構造函數
BigInt("9007199254740995"); // → 9007199254740995n
複製代碼
BigInt
文字也能夠用二進制、八進制或十六進制表示
// binary
console.log(0b100000000000000000000000000000000000000000000000000011n);
// → 9007199254740995n
// hex
console.log(0x20000000000003n);
// → 9007199254740995n
// octal
console.log(0o400000000000000003n);
// → 9007199254740995n
// note that legacy octal syntax is not supported
console.log(0400000000000000003n);
// → SyntaxError
複製代碼
請記住,不能使用嚴格相等運算符將BigInt
與常規數字進行比較,由於它們的類型不一樣:
console.log(10n === 10); // → false
console.log(typeof 10n); // → bigint
console.log(typeof 10); // → number
複製代碼
相反,可使用等號運算符,它在處理操做數以前執行隱式類型轉換
console.log(10n == 10); // → true
複製代碼
除一元加號(+
)運算符外,全部算術運算符均可用於BigInt
10n + 20n; // → 30n
10n - 20n; // → -10n
+10n; // → TypeError: Cannot convert a BigInt value to a number
-10n; // → -10n
10n * 20n; // → 200n
20n / 10n; // → 2n
23n % 10n; // → 3n
10n ** 3n; // → 1000n
const x = 10n;
++x; // → 11n
--x; // → 9n
複製代碼
不支持一元加號(+
)運算符的緣由是某些程序可能依賴於+
始終生成Number
的不變量,或者拋出異常。 更改+
的行爲也會破壞asm.js
代碼。
固然,與BigInt
操做數一塊兒使用時,算術運算符應該返回BigInt
值。所以,除法(/
)運算符的結果會自動向下舍入到最接近的整數。例如:
25 / 10; // → 2.5
25n / 10n; // → 2n
複製代碼
由於隱式類型轉換可能丟失信息,因此不容許在bigint
和 Number
之間進行混合操做。當混合使用大整數和浮點數時,結果值可能沒法由BigInt
或Number
精確表示。思考下面的例子:
(9007199254740992n + 1n) + 0.5
複製代碼
這個表達式的結果超出了BigInt
和Number
的範圍。小數部分的Number
不能精確地轉換爲BigInt
。大於2^53
的BigInt
不能準確地轉換爲數字。
因爲這個限制,不可能對混合使用Number
和BigInt
操做數執行算術操做。還不能將BigInt
傳遞給Web api和內置的 JS 函數,這些函數須要一個 Number
類型的數字。嘗試這樣作會報TypeError
錯誤
10 + 10n; // → TypeError
Math.max(2n, 4n, 6n); // → TypeError
複製代碼
請注意,關係運算符不遵循此規則,以下例所示:
10n > 5; // → true
複製代碼
若是但願使用BigInt
和Number
執行算術計算,首先須要肯定應該在哪一個類型中執行該操做。爲此,只需經過調用Number()
或BigInt()
來轉換操做數:
BigInt(10) + 10n; // → 20n
// or
10 + Number(10n); // → 20
複製代碼
當 Boolean
類型與BigInt
類型相遇時,BigInt
的處理方式與Number
相似,換句話說,只要不是0n
,BigInt
就被視爲truthy
的值:
if (5n) {
// 這裏代碼塊將被執行
}
if (0n) {
// 這裏代碼塊不會執行
}
複製代碼
排序BigInts
和Numbers
數組時,不會發生隱式類型轉換:
const arr = [3n, 4, 2, 1n, 0, -1n];
arr.sort(); // → [-1n, 0, 1n, 2, 3n, 4]
複製代碼
位操做符如|、&、<<、>>
和^
對Bigint
的操做方式與Number
相似。下面是一些例子
90 | 115; // → 123
90n | 115n; // → 123n
90n | 115; // → TypeError
複製代碼
與其餘基本類型同樣,可使用構造函數建立BigInt
。傳遞給BigInt()
的參數將自動轉換爲BigInt
:
BigInt("10"); // → 10n
BigInt(10); // → 10n
BigInt(true); // → 1n
複製代碼
沒法轉換的數據類型和值會引起異常:
BigInt(10.2); // → RangeError
BigInt(null); // → TypeError
BigInt("abc"); // → SyntaxError
複製代碼
能夠直接對使用構造函數建立的BigInt
執行算術操做
BigInt(10) * 10n; // → 100n
複製代碼
使用嚴格相等運算符的操做數時,使用構造函數建立的Bigint
與常規Bigint
的處理方式相似
BigInt(true) === 1n; // → true
複製代碼
在撰寫本文時,Chrome +67
和Opera +54
徹底支持BigInt
數據類型。不幸的是,Edge
和Safari
尚未實現它。Firefox
默認不支持BigInt,可是能夠在about:config
中將javascript.options.bigint
設置爲true
來開啓它,最新支持的狀況可在「Can I use」上查看。
不幸的是,轉換BigInt
是一個極其複雜的過程,這會致使嚴重的運行時性能損失。直接polyfill BigInt
也是不可能的,由於該提議改變了幾個現有操做符的行爲。目前,更好的選擇是使用JSBI庫,它是BigInt
提案的純JS實現。
這個庫提供了一個與原生BigInt
行爲徹底相同的API。下面是如何使用JSBI:
import JSBI from './jsbi.mjs';
const b1 = JSBI.BigInt(Number.MAX_SAFE_INTEGER);
const b2 = JSBI.BigInt('10');
const result = JSBI.add(b1, b2);
console.log(String(result)); // → '9007199254741001'
複製代碼
使用JSBI
的一個優勢是,一旦瀏覽器支持,就不須要重寫代碼。 相反,可使用babel
插件自動將JSBI代碼編譯爲原生 BigInt
代碼。
BigInt
是一種新的數據類型,用於當整數值大於Number
數據類型支持的範圍時。這種數據類型容許咱們安全地對大整數執行算術操做,表示高分辨率的時間戳,使用大整數id,等等,而不須要使用庫。
重要的是要記住,不能使用Number
和BigInt
操做數的混合執行算術運算,須要經過顯式轉換其中的一種類型。 此外,出於兼容性緣由,不容許在BigInt
上使用一元加號(+
)運算符。
阿里雲最近在作活動,低至2折,有興趣能夠看看:promotion.aliyun.com/ntms/yunpar…
乾貨系列文章彙總以下,以爲不錯點個Star,歡迎 加羣 互相學習。
我是小智,公衆號「大遷世界」做者,對前端技術保持學習愛好者。我會常常分享本身所學所看的乾貨,在進階的路上,共勉!
關注公衆號,後臺回覆福利,便可看到福利,你懂的。
每次整理文章,通常都到2點才睡覺,一週4次左右,挺苦的,還望支持,給點鼓勵