目次
二、浮点数运算的精度丢失
三、大整数的运算精度丢失及溢出
ホームページ ウェブフロントエンド jsチュートリアル JavaScript の Number 型に関するよくある誤解の背後にある原則と解決策

JavaScript の Number 型に関するよくある誤解の背後にある原則と解決策

Oct 09, 2022 am 11:56 AM
javascript

この記事では、JavaScript に関する関連知識をお届けします。主に数値型に関する関連知識を紹介します。数値型に関するよくある誤解の背後にある原則や解決策なども含まれます。次のとおりです。見てください、それが皆さんのお役に立てば幸いです。

JavaScript の Number 型に関するよくある誤解の背後にある原則と解決策

[関連する推奨事項: JavaScript ビデオ チュートリアル Web フロントエンド ]

JavaScript では、 value は 1 つの Type、つまり Number 型のみで、内部的には倍精度浮動小数点型、つまり他の言語の double 型として表されるため、実際には JavaScript には整数型は存在しません。浮動小数点数として処理され、格納方法は IEEE 754 国際標準に従って同じです。したがって、JavaScript では 3 と 3.0 は同じ値として扱われます。

3.0 === 3 // true
ログイン後にコピー

整数の場合、正確に計算できる整数の範囲は、−253-2^{53}−253 ~ 2532^{53 です。 253 の間 (2 つのエンドポイントを除く)、この範囲内にある限り、整数を安全に使用できます。 10 進数に加えて、整数は 8 進数または 16 進数のリテラル値でも表すことができます。8 進数のリテラル値の最初の桁は 0 であり、その後に一連の 8 進数 (0 ~ 7) が続く必要があります。リテラル値の値が範囲外の場合、先頭のゼロは無視され、次の値は 10 進数として解析されます。厳密モードでは、この 8 進数表現はエラーを報告することに注意してください。ES6 では、8 進数表現が必要であることをさらに明確にしています。プレフィックス 0o を使用します。例:

(function(){
  console.log(0o11 === 011)
})()
// true
// 严格模式
(function(){
  'use strict';
  console.log(0o11 === 011)
})()
// Uncaught GyntaxError
ログイン後にコピー

16 進リテラル値の最初の 2 桁は 0x で、その後に任意の 16 進数字 (0 ~ 9 および A ~ F) が続きます。A ~ F は大文字または小文字。 ES6 では、プレフィックス 0b (または 0B) を使用するバイナリ書き込み方法が拡張されました。

先ほど、JavaScript ランタイムの Number 型について簡単に紹介し、その後、これらの一般的な問題を正式に紹介し始めましたが、その前に、Number 型のデータ格納方法を理解する必要があります。

1. 格納方法

JavaScript の Number 型は、他の言語では double 型である倍精度浮動小数点を使用します。倍精度浮動小数点数は、格納に 8 バイト (64 ビット) を使用します。 . 最新のコンピュータの浮動小数点数 浮動小数点数のほとんどは、国際標準 IEEE 754 に従って保存されます。保存プロセスは 2 つのステップに分かれています。

  • 浮動小数点数を対応する浮動小数点数に変換します。 2 進数を科学表記法で表現します

  • 変換された数値を、IEEE 754 標準を通じて実際にコンピュータに保存される値に表します。

#IEEE 754 標準によれば、任意の 2 進浮動小数点数 V は次のように表現できます。

JavaScript の Number 型に関するよくある誤解の背後にある原則と解決策

JavaScript の Number 型に関するよくある誤解の背後にある原則と解決策# #たとえば、10 進数の 5.0 は 2 進数で表記すると 101.0 であり、これは 1.01*221.01 * 2^21.01*22 (S=0、M=1.01、E=2) に相当します。

IEEE 754 では、以下に示すように、32 ビット浮動小数点数について、最上位桁が符号ビット S、次の 8 ビットが指数 E、残りの 23 ビットが有効桁 M であると規定しています。次の図:

JavaScript の Number 型に関するよくある誤解の背後にある原則と解決策64 ビット浮動小数点数の場合、最上位の桁は符号ビット S、次の 11 ビットは指数 E、残りの 11 ビットは符号ビット S です。次の図に示すように、52 ビットが有効数字 M です。

JavaScript の Number 型に関するよくある誤解の背後にある原則と解決策注: IEEE754 には、有効数字 M と指数 E に関する特別な規則もあります。

前に述べたように、1 <= M < 2、つまり M は常に 1.xxxxxxx の形式で記述できます。xxxxxxx は効率的な部分を表します。 IEEE 754 では、M をコンピュータに保存する場合、デフォルトではこの数値の最初の桁は常に 1 であるため、これを破棄して次の xxxxxxx 部分のみを保存することが規定されています。たとえば、1.01 を保存する場合、01 だけが保存され、読み出し時に最初の 1 が追加されます。これは有効桁 1 桁を保存するためです。32 ビット浮動小数点数を例にとると、有効桁 M は次のようになります。 left. 23 桁しかありません。最初の 1 を四捨五入すると、有効数字 24 桁を節約することになります。

インデックス E に関しては、状況はさらに複雑です。まず、E は符号なしの指数です。これは、E が 8 ビットの場合、その値の範囲は 0 ~ 255 であり、E が 11 ビットの場合、その値の範囲は 0 ~ 2047 であることを意味します。しかし、科学表記法における E は負の値を持つ可能性があることがわかっているため、IEEE 754 では、E の実数値を中間数値から減算する必要があると規定しています。8 桁の E の場合、この中間数値は 127 であり、11 桁の場合は、この中間数値は 127 になります。 E 、中央の数字は 1023 です。

たとえば、2102^{10}210 の E は 10 であるため、32 ビット浮動小数点数として保存する場合は、10 127=137、つまり 10001001 として保存する必要があります。

次に、インデックス E は 3 つの状況に分類できます:

  • E不全为0或不全为1:这时,浮点数就采用上面的规则表示,即指数E的计算值减去127(或1023),得到真实值,再将有效数字M前加上第一位的1。

  • E全为0:这时,浮点数的指数E等于1 ~ 127(或1 ~ 1023),有效数字M不再加上第一位的1,而是还原成0.xxxxxxx的小数,这样做是为了表示±0,以及接近0的很小的数字。

  • E全为1:这时,如果有效数字M全为0,表示±无穷大(正负取决于符号位S);如果有效数字M不全为0,表示这个数不是一个数(NaN)。

示例:浮点数 9.0 如何用二进制表示?还原成十进制又是多少?

首先,浮点数 9.0 等于二进制的 1001.0,即 1.001∗231.001 *2^31.001∗23

那么,第一位的符号位 S=0,有效数字 M 等于 001 后面再加 20 个 0,凑满 23 位,指数 E 等于 3+127=130,即 10000010。

所以,写成二进制形式,应该是 S+E+M,即0 10000010 001 0000 0000 0000 0000 0000。这个 32 位的二进制数,还原成十进制,正是 1091567616。

注:虽然在 JavaScript 中无论是小数还是整数都是按照64位的浮点数形式存储,但是进行整数运算会自动转换为32位的有符号整数,例如位运算,有符号整数使用31位表示整数的数值,用第32位表示整数的符号,数值范围是−231-2^{31}−231 ~ 2312^{31}231。

二、浮点数运算的精度丢失

问题缘由

众所周知在 JavaScript 中 0.1+0.2 不等于 0.3,实际上所有浮点数值存储遵循 IEEE 754 标准的编程语言中都会存在这个问题,这是因为计算机中小数的存储先是转换成二进制进行存储的,而 0.1、0.2 转换成二进制分别为:

(0.1)10 => (00011001100110011001(1001)...)2 (0.2)10 => (00110011001100110011(0011)...)2

可以发现,0.1 和 0.2 转成二进制之后都是一个无限循环的数,前面提到尾数位只能存储最多 53 位有效数字,这时候就必须来进行四舍五入了,而这个取舍的规则就是在 IEEE 754 中定义的,0.1 最终能被存储的有效数字是

0001(1001)(1001)(1001)(1001)(1001)(1001)(1001)(1001)(1001)(1001)(1001)(1001)101 + (0011)(0011)(0011)(0011)(0011)(0011)(0011)(0011)(0011)(0011)(0011)(0011)(0011)01 = 0100(1100)(1100)(1100)(1100)(1100)(1100)(1100)(1100)(1100)(1100)(1100)(1100)111

最终的这个二进制数转换成十进制的就是 0.30000000000000004 ,这儿需要注意,53 位的存储位指的是能存 53 位有效数字,因此前置的 0 不算,要往后再取到 53 位有效数字为止。

因此,精度丢失的问题实际上用一句话概括就是计算机中用二进制存储小数,而大部分小数转成二进制后都是无限循环的值,因此存在取舍问题,也就是精度丢失。

解决办法

ES6 在 Number 对象上新增了一个极小常量:Number.EPSILON,值为 2.220446049250313e-16,引入这么一个常量就是为了为浮点数计算设置一个误差范围,如果这个误差小于 Number.EPSILON 我们就认为得到了准确结果。

三、大整数的运算精度丢失及溢出

问题缘由

在介绍问题的具体缘由之前我想先给大家介绍一下所谓最大安全整数范围以及最大数字绝对值的范围是如何得到的?

JavaScript 能够表示的数字的绝对值范围是 5e-324 ~ 1.7976931348623157e+308,这两个取值可以通过 Number.MIN_VALUE 和 Number.MAX_VALUE 这两个字段来表示,如果某次计算的结果得到了一个超出 JavaScript 数值范围的,那么这个数值会自动被转换为特殊的 Infinity 值,具体来说,如果这个数是负数,则会被转换成 -Infinity(负无穷),如果这个数值是正数,则会被转换成 Infinity(正无穷)。

示例:

console.log(Number.MAX_VALUE) // 1.7976931348623157e+308
console.log(Number.MIN_VALUE) // 5e-324
console.log(Number.MAX_VALUE + Number.MAX_VALUE) // Infinity
ログイン後にコピー

那么这个取值范围是如何得到的呢?

前面说到 JavaScript 中数值的保存采用的是双精度浮点型,遵循 IEEE 754 标准,在 ECMAScript 规范中规定指数 E 的范围在 -1074 ~ 971,双精度浮点型中有效数字 M 的存储位为52,但是有效数字 M 由于可以省略第一位1,节省一个存储位,因此有效数字M可以存储的范围为 1 ~ 2532^{53}253,因此 JavaScript 中 Number 能表示的最大数字绝对值范围是 2−10742^{-1074}2−1074 ~ 253+9712^{53+971}253+971。

注:通过 Number.isFinite()(ES6引入)和 isFinite() 方法可以判断一个数值是不是有穷的,即如果参数位于最小与最大数值之间时会返回 true。

让我们回归主题,为什么会出现大整数的运算精度丢失及溢出呢?

JavaScript 中最大安全整数的范围是 −253-2^{53}−253 ~ 2532^{53}253,不包括两个端点,即 -9007199254740991 ~ 9007199254740991,可以通过 Number.MIN_SAFE_INTEGER 和 Number.MAX_SAFE_INTEGER 字段查询,超出这个范围的整数计算都是不准确的,例如:

console.log(Number.MAX_SAFE_INTEGER) // 9007199254740991
console.log(Number.MIN_SAFE_INTEGER) // -9007199254740991
console.log(9007199254740991 + 2) // 9007199254740992
ログイン後にコピー

最大安全整数9007199254740991对应的二进制数如图:

JavaScript の Number 型に関するよくある誤解の背後にある原則と解決策

53位有效数字都存储满了之后,想要表示更大的数字,就只能往指数数加一位,这时候尾数因为没有多余的存储空间,因此只能补0。

JavaScript の Number 型に関するよくある誤解の背後にある原則と解決策

如图所示,在指数位为53的情况下,最后一位尾数位为0的数字可以被精确表示,而最后一位尾数位为1的数字都不能被精确表示。也就是可以被精确表示和不能被精确表示的比例是1:1。

同理,当指数为54的时候,只有最后两位尾数为00的可以被精确表示,也就是可以被精确表示和不能被精确表示的比例是1:3,当有效位数达到 x(x>53) 的时候,可以被精确表示和不能被精确表示的比例将是1 : 2^(x-53)^ - 1。

可以预见的是,在指数越来越高的时候,这个指数会成指数增长,因此在 Number.MAX_SAFE_INTEGER ~ Number.MAX_VALUE 之间可以被精确表示的整数可以说是凤毛麟角。

之所以会有最大安全整数这个概念,本质上还是因为数字类型在计算机中的存储结构。在尾数位不够补零之后,只要是多余的尾数为1所对应的整数都不能被精确表示。

可以发现,不管是浮点数计算的计算结果错误和大整数的计算结果错误,最终都可以归结到JS的精度只有53位(尾数只能存储53位的有效数字)。

解决办法

那么我们在日常工作中碰到这两个问题该如何解决呢?

大而全的解决方案就是使用 mathjs,看一下 mathjs 的输出:

math.config({
    number: &#39;BigNumber&#39;,      
    precision: 64 
});
console.log(math.format(math.eval(&#39;0.1 + 0.2&#39;))); // &#39;0.3&#39;
console.log(math.format(math.eval(&#39;0.23 * 0.34 * 0.92&#39;))); // &#39;0.071944&#39;
console.log(math.format(math.eval(&#39;9007199254740991 + 2&#39;))); 
// &#39;9.007199254740993e+15&#39;
ログイン後にコピー

【相关推荐:JavaScript视频教程web前端

以上がJavaScript の Number 型に関するよくある誤解の背後にある原則と解決策の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

AI Hentai Generator

AI Hentai Generator

AIヘンタイを無料で生成します。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

WebSocket と JavaScript を使用してオンライン音声認識システムを実装する方法 WebSocket と JavaScript を使用してオンライン音声認識システムを実装する方法 Dec 17, 2023 pm 02:54 PM

WebSocket と JavaScript を使用してオンライン音声認識システムを実装する方法 はじめに: 技術の継続的な発展により、音声認識技術は人工知能の分野の重要な部分になりました。 WebSocket と JavaScript をベースとしたオンライン音声認識システムは、低遅延、リアルタイム、クロスプラットフォームという特徴があり、広く使用されるソリューションとなっています。この記事では、WebSocket と JavaScript を使用してオンライン音声認識システムを実装する方法を紹介します。

WebSocket と JavaScript: リアルタイム監視システムを実装するための主要テクノロジー WebSocket と JavaScript: リアルタイム監視システムを実装するための主要テクノロジー Dec 17, 2023 pm 05:30 PM

WebSocketとJavaScript:リアルタイム監視システムを実現するためのキーテクノロジー はじめに: インターネット技術の急速な発展に伴い、リアルタイム監視システムは様々な分野で広く利用されています。リアルタイム監視を実現するための重要なテクノロジーの 1 つは、WebSocket と JavaScript の組み合わせです。この記事では、リアルタイム監視システムにおける WebSocket と JavaScript のアプリケーションを紹介し、コード例を示し、その実装原理を詳しく説明します。 1.WebSocketテクノロジー

JavaScript と WebSocket を使用してリアルタイムのオンライン注文システムを実装する方法 JavaScript と WebSocket を使用してリアルタイムのオンライン注文システムを実装する方法 Dec 17, 2023 pm 12:09 PM

JavaScript と WebSocket を使用してリアルタイム オンライン注文システムを実装する方法の紹介: インターネットの普及とテクノロジーの進歩に伴い、ますます多くのレストランがオンライン注文サービスを提供し始めています。リアルタイムのオンライン注文システムを実装するには、JavaScript と WebSocket テクノロジを使用できます。 WebSocket は、TCP プロトコルをベースとした全二重通信プロトコルで、クライアントとサーバー間のリアルタイム双方向通信を実現します。リアルタイムオンラインオーダーシステムにおいて、ユーザーが料理を選択して注文するとき

WebSocketとJavaScriptを使ったオンライン予約システムの実装方法 WebSocketとJavaScriptを使ったオンライン予約システムの実装方法 Dec 17, 2023 am 09:39 AM

WebSocket と JavaScript を使用してオンライン予約システムを実装する方法 今日のデジタル時代では、ますます多くの企業やサービスがオンライン予約機能を提供する必要があります。効率的かつリアルタイムのオンライン予約システムを実装することが重要です。この記事では、WebSocket と JavaScript を使用してオンライン予約システムを実装する方法と、具体的なコード例を紹介します。 1. WebSocket とは何ですか? WebSocket は、単一の TCP 接続における全二重方式です。

JavaScript と WebSocket: 効率的なリアルタイム天気予報システムの構築 JavaScript と WebSocket: 効率的なリアルタイム天気予報システムの構築 Dec 17, 2023 pm 05:13 PM

JavaScript と WebSocket: 効率的なリアルタイム天気予報システムの構築 はじめに: 今日、天気予報の精度は日常生活と意思決定にとって非常に重要です。テクノロジーの発展に伴い、リアルタイムで気象データを取得することで、より正確で信頼性の高い天気予報を提供できるようになりました。この記事では、JavaScript と WebSocket テクノロジを使用して効率的なリアルタイム天気予報システムを構築する方法を学びます。この記事では、具体的なコード例を通じて実装プロセスを説明します。私たちは

JavaScriptでinsertBeforeを使用する方法 JavaScriptでinsertBeforeを使用する方法 Nov 24, 2023 am 11:56 AM

使用法: JavaScript では、insertBefore() メソッドを使用して、DOM ツリーに新しいノードを挿入します。このメソッドには、挿入される新しいノードと参照ノード (つまり、新しいノードが挿入されるノード) の 2 つのパラメータが必要です。

簡単な JavaScript チュートリアル: HTTP ステータス コードを取得する方法 簡単な JavaScript チュートリアル: HTTP ステータス コードを取得する方法 Jan 05, 2024 pm 06:08 PM

JavaScript チュートリアル: HTTP ステータス コードを取得する方法、特定のコード例が必要です 序文: Web 開発では、サーバーとのデータ対話が頻繁に発生します。サーバーと通信するとき、多くの場合、返された HTTP ステータス コードを取得して操作が成功したかどうかを判断し、さまざまなステータス コードに基づいて対応する処理を実行する必要があります。この記事では、JavaScript を使用して HTTP ステータス コードを取得する方法を説明し、いくつかの実用的なコード例を示します。 XMLHttpRequestの使用

JavaScript と WebSocket: 効率的なリアルタイム画像処理システムの構築 JavaScript と WebSocket: 効率的なリアルタイム画像処理システムの構築 Dec 17, 2023 am 08:41 AM

JavaScript は Web 開発で広く使用されているプログラミング言語であり、WebSocket はリアルタイム通信に使用されるネットワーク プロトコルです。 2 つの強力な機能を組み合わせることで、効率的なリアルタイム画像処理システムを構築できます。この記事では、JavaScript と WebSocket を使用してこのシステムを実装する方法と、具体的なコード例を紹介します。まず、リアルタイム画像処理システムの要件と目標を明確にする必要があります。リアルタイムの画像データを収集できるカメラ デバイスがあるとします。

See all articles