跨浏览器Javascript数字精度
在 JavaScript 中,数字被定义为 64 位双精度。我对分布式 Web 应用程序有一个特定的用途,只有当我可以依赖所有浏览器的一致结果时,它才有效。
尽管规范使用 IEEE 标准,但我自然怀疑数学库甚至底层硬件的实现可能存在微小差异,这可能会导致复合错误。
是否有任何兼容性数据来源或可靠的测试套件来验证浏览器中的双精度计算?特别是,我还需要考虑移动浏览器(通常基于 ARM)。
澄清 -
这是一个有关浏览器兼容性的问题。我试图了解是否所有浏览器都可以按照 IEEE 浮点定义的可靠、一致和可重复的方式处理数字。在大多数语言中,这是一个安全的假设,但有趣的是,在浏览器中对此存在一些不确定性。
关于如何避免由于缺乏精度和舍入误差而导致的浮点问题,有一些很好的建议。在大多数情况下,如果您需要准确性,您应该遵循此建议!
对于这个问题,我并不是要回避问题,而是要理解它。浮点数在设计上本质上是不准确的,但只要注意构建的方式,不准确度是完全可以预测和一致的。 IEEE-754 对此的描述达到了只有标准机构才能做到的详细程度。
我决定提供一笔小额赏金
- 如果有人能引用与主流浏览器中 IEEE 编号实现相关的真实兼容性数据,
- 。测试套件旨在验证浏览器内的实现,包括验证 64 位浮点数(53 位尾数)的正确内部使用。
在这个问题中,我并不是在寻找替代选项、解决方法或避免问题的方法。谢谢你的建议。
Within JavaScript, numbers are defined as 64bit double-precision. I have a specific use in mind for a distributed web application, which would only work if I can rely on consistent results across all browsers.
Despite the spec using the IEEE standard, I naturally have a suspicion that there may be tiny differences in implementations of the maths library or even the underlying hardware, which could cause compound errors.
Is there any source of compatibility data, or a reliable test suite to verify double precision calculations in the browser? In particular, I also need to consider mobile browsers (usually ARM based).
Clarification -
This is a question about browser compatibility. I'm trying to understand whether all browsers can be relied upon to treat numbers in a reliable, consistent and repeatable way as defined for IEEE floating point. In most languages this is a safe assumption, but it's interesting that there's a little uncertainty about this in the browser.
There's been some great advice on how to avoid floating point problems due to lack of precision and rounding errors. In most cases, if you require accuracy you should follow this advice!
For this question, I'm not trying to avoid the problem but understand it. Floating point numbers are inherently inaccurate by design, but as long as some care is taken with how builds are made that inaccuracy can be completely predictable and consistent. IEEE-754 describes this to a level of detail that only a standards body could.
I've decided to offer a small bounty if anyone can cite,
- Genuine compatibility data relating to the implementation of IEEE numbers in mainstream browsers.
- A test suite intended to verify the implementation within the browsers, including verifying the correct internal use of a 64 bit floating point number (53 bit mantissa).
In this question I'm not looking for alternative options, workarounds or ways to avoid the problem. Thank you for the suggestions.
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(7)
这只是为了好玩,正如您已经说过的,我创建了一个新答案,因为这个答案是不同的。但我仍然觉得有一些路人忽视了这个问题的无用性。因此,让我们首先阐述您的观点:
首先:
不存在,就此而言甚至没有任何意义, IEEE 只是一个标准机构...?我不确定这种模糊是有意还是无意,我假设您试图说 IEEE 754,但问题就在这里......技术上该标准有 2 个版本 IEEE 754-2008 和 IEEE 754-1985。基本上前者是较新的并且解决了后者的疏忽。任何理智的人都会认为任何维护的 JavaScript 实现都会更新到最新和最好的标准,但任何理智的人都应该比这更好地了解 JavaScript,即使 JavaScript 并不疯狂,也没有规范说该实现必须是/保持最新(如果您不相信我,他们甚至不谈论“版本”)。更复杂的是,IEEE 浮点运算标准 754-2008 支持两种
编码格式:十进制编码格式、二进制编码格式。正如预期的那样,它们是相互兼容的,因为您可以来回移动而不会丢失数据,但这是假设我们可以访问数字的二进制表示形式,但我们不能 (无需附加调试器并以老式方式查看存储)
但是,据我所知,通常的做法是用老式的
“返回”JavaScript
这当然意味着我们处于取决于用于实际构建浏览器的编译器。但即使在这个领域,我们也不能也不应该假设平等,即使所有编译器都使用同一版本的标准(它们不是),即使所有编译器完整地实现了该标准(它们不)。以下是这篇论文的摘录,我认为这篇论文有趣、有价值且相关-此对话框读...Number
double同时发现我还发现了这个完全用 JavaScript 完成的参考实现(注意:我实际上还没有验证实施的有效性)。
说了这么多,让我们继续讨论您的第二个请求:
由于 JavaScript 是一个解释平台,您现在应该看到,没有办法以绝对可靠的方式测试脚本 + 编译器(VM/引擎)+ 编译编译器 + 机器的编译器的集合 JavaScript 的重点。因此,除非您想构建一个充当浏览器主机的测试套件,并实际上“窥视”进程的私有内存以确保有效的表示,否则这很可能是徒劳的,因为该数字很可能是由一个
double
,这将符合浏览器内置的 C 或 C++ 中的要求。 JavaScript 中没有绝对的方法可以做到这一点,因为我们只能访问“对象”即使我们查看Number
在控制台中,我们正在查看.toString
版本。就此而言,我认为这是唯一重要的形式,因为它将从二进制文件中确定,并且只有在语句:n1 === n2 && 时才会成为失败点。 n1.toString() !== n2.toString()
你可以找到一个相关的n1, n2
...也就是说,我们可以测试字符串版本,实际上它只要我们记住一些奇怪的地方,就和测试二进制文件一样好。特别是因为 JavaScript 引擎/VM 之外的任何东西都不会触及二进制版本。然而,这会让你受到一个奇怪的、特定的、可能非常挑剔的、随时可能改变的故障点的摆布。仅供参考,这里摘录自 webkit 的 JavaScriptCore 的 Number Prototype (NumberPrototype.cpp) 显示转换的复杂性:
...如您所见,这里的数字由传统的
double
支持,并且转换为绝非简单明了。所以我的设计是这样的:因为我推测这些实现的唯一不同之处是它们对字符串的“渲染”。我构建了一个包含三部分的测试生成器:为了实现这一目标,我们需要访问参考构建,我的第一个想法是使用本地语言的参考构建,但我发现生成的数字似乎比 JavaScript 具有更高的精度,通常会导致更多错误。然后我想,如果我只使用 JavaScript 引擎中已有的实现会怎样。 WebKit/JavaScriptCore 似乎是一个非常好的选择,但构建和运行参考也需要做很多工作,所以我选择了 .NET 的简单性,因为它可以访问“jScript”,虽然最初看起来并不理想考试产生比本地同行更接近的结果。我真的不想用 jScript 进行编码,因为该语言几乎已被弃用,所以我选择通过
CodeDomProvider
使用 C# 引导 jScript。经过一番修改后,它生成了以下内容:http://jsbin.com/afiqil (最后是演示酱!!!1!),所以现在您可以在所有浏览器中运行它并编译您自己的数据,根据我个人的检查,我尝试过的每个浏览器中的字符串舍入解释似乎有所不同,但是我还没有找到一个主要的浏览器来处理幕后的数字(除了stringify-ing)不同...现在是 C# 酱汁:
或者预编译版本(如果您应该选择): http://uploading.com/files/ad4a85md/DoubleFloatJs.exe/ 这是一个可执行文件,下载风险自负
我应该提到该程序的目的只是在文本框中生成一个 JavaScript 文件并将其复制到剪贴板以供使用方便为了粘贴到您选择的任何地方,您可以轻松地将其翻转并将其放在 asp.net 服务器上,并将报告添加到结果中以 ping 服务器并在某些大型数据库中进行跟踪...这就是我会做的,如果我想要信息..
...并且,...我,...花了我希望这对您有帮助 -ck
This is just for fun, as you already stated and I created a new answer because this one is in a different vein. But I still feel like there are a few random passerby's who are ignoring the futility of the problem. So let's start by addressing your points:
Firstly:
doesn't exist, and for that matter doesn't even make any sense, IEEE is just a standards body...? I am not sure if this vague on purpose or on accident, I will assume you were trying to say IEEE 754, but there in lies the rub... there are technically 2 versions of this standard IEEE 754-2008 AND IEEE 754-1985. Basically the former is newer and addresses the latter's oversights. Any sane person would assume that any maintained JavaScript implementation would update to the latest and greatest standard, but any sane person should know JavaScript better than that, and even if JavaScript wasn't crazy, there is no specification saying that the implementation has to be/stay up to date (check the ECMA spec yourself if you don't believe me, they don't even talk "versions"). To compound the matters further the IEEE Standard 754-2008 for Floating-Point Arithmetic supports two
encoding formats: the decimal encoding format, and the binary encoding format. Which as would be expected are compatible with each other in the sense that you can go back and forth without loss of data, but that's assuming we have access to the binary representation of the number, which we don't (without attaching a debugger and looking at the store the old school way)
However, from what I can tell it seems it is general practice to "back" a JavaScript
Number
with an old fashioneddouble
which of course means that we are at the mercy of the compiler used to actually build the browser. But even in that realm, we can't and shouldn't be assuming equality even if all the compilers were on the same version of the standard (they aren't) and even if all the compilers implemented the standard in its entirety (they don't). Here's an excerpt from this paper, which I have deemed an interesting, worthwhile and relevant-to-this-dialog read...While finding that I also found this reference implementation done completely in JavaScript (note: I haven't actually verified the validity of the implementation).
All that said, let's move on to your second request:
Since JavaScript is an interpreted platform you should see now that there is no way to test the set of script + compiler (VM/engine) + compiler that compiled the compiler + machine in an absolute and reliable way from the point of JavaScript. So unless you want to build a test suite that acts as a browser host and actually "peeks" into the private memory of the process to ensure a valid representation, which would be fruitless most likely anyway since the number are most likely "backed" by a
double
and that is going to conform as it does in the C or C++ that browser was built in. There is no absolute way to do this from JavaScript since all we have access to is the "object" and even when we view theNumber
in a console we are looking at a.toString
version. For that matter I would posit that this is the only form that matters since it will be determined from the binary and would only become a point of failure if for the statement:n1 === n2 && n1.toString() !== n2.toString()
you could find ann1, n2
that is relevant...That said, we can test the string version and in reality it is just as good as testing the binary as long as we keep a few oddities in mind. Especially since nothing outside the JavaScript engine/VM ever touches the binary version. However this puts you at the mercy of an oddly specific, possibly very finicky and poised to be changed point of failure. Just for reference, here is an excerpt from webkit's JavaScriptCore's Number Prototype (NumberPrototype.cpp) displaying the complexity of the conversion:
... as you can see, the number here is backed by a traditional
double
and the conversion is anything but simple and straightforward. So what I devised was this: since I conjecture that the only spot that these implementations will differ are their "rendering" to strings. I built a test generator that is three fold:To accomplish this we need access to a reference build, my first thought was to use one from a native language but with that I found that the numbers produced seemed to have a higher precision than JavaScript in general leading to far more errors. So then I thought, what if I just used an implementation already inside a JavaScript engine. WebKit/JavaScriptCore seemed like a really good choice but it would have also been a lot of work to get the reference build up and running so I opted for the simplicity of .NET since it has access to "jScript" while not ideal seemed upon initial examination to produce closer results than the native counterpart. I didn't really want to code in jScript since the language is all but deprecated so I opted for C# bootstrapping jScript through a
CodeDomProvider
.... After a little tinkering here's what it produced: http://jsbin.com/afiqil (finally demo sauce!!!!1!), so now you can run it in all browsers and compile your own data, which upon my personal inspection it seems string rounding interpretation varies in EVERY browser I tried, however I've yet to find a major browser that handled the numbers behind the scenes (other that the stringify-ing) differently...now for the C# sauce:
or a pre-compiled version if you should choose: http://uploading.com/files/ad4a85md/DoubleFloatJs.exe/ this is an executable, download at your own risk
I should mention that the purpose of the program is just to produce a JavaScript file in a text box and copy it to the clipboard for convenience for pasting wherever you choose, you could easily turn this around and put it on an asp.net server and add reporting to results to ping the server and keep track in some massive database... which is what I would do to it if I desired the information..
...and, ...I'm, ...spent I hope this helps you -ck
总结下面的所有内容,除了一些 IE 的故障之外,您可以期望大多数系统都符合要求,但应该使用健全性检查作为预防措施(包括建议)。
ECMAScript 规范(均修订版) 3 和 5)对于 IEEE 754 的特性非常精确,例如转换、舍入模式、上溢/下溢,甚至有符号零(第 5.2、8.5、9.* 节) 11.5.*、11.8.5;15.7 和 15.8 也处理浮点数)。它不仅仅是“将事情留给底层实现”。 vv 之间没有明显差异。 3 和 5 以及所有主流浏览器至少支持 v.3。因此,每个人都遵守其规则,至少名义上如此。让我们看看...
没有浏览器完全通过 test262 ECMAScript 合规性测试 (WP#一致性测试)。但是,没有 test262 错误 在 google 上找到 是与浮动相关。
IE5.5+ ( [MS-ES3])报告
Number.toFixed
、Number.toExponential
中的差异,Number.toPrecision
。其他差异与浮动无关。我无法在 IE8 中运行 test262;要验证系统,您可以使用 test262 中与浮动相关的测试。它们位于
http://test262.ecmascript.org/json/ch<规范章节的 2 位数字>.json
;可以使用(python 2.6+)提取测试代码:另一个机会是 C 语言的 IEEE 754 合规性测试。
test262 的相关部分(比较浮点数的部分)如下:
可以在此处找到此列表和所有相关测试文件的串联源(截至 2012 年 3 月 9 日,没有来自该工具的文件): http://pastebin.com/U6nX6sKL
Summarizing everything below, you can expect compliance on the majority of systems save for a few IE's glitches, but ought to use a sanity check as a precaution (proposition is included).
ECMAScript specification (both rev. 3 and 5) is very precise on IEEE 754's peculiarities such as conversions, rounding mode, overflow/underflow and even signed zeros (sec. 5.2, 8.5, 9.*, 11.5.*, 11.8.5; 15.7 and 15.8 deal with floats too). It doesn't just "leave things to the underlying implementation". There are no apparent differences between vv. 3 and 5 and all major browsers support v.3 at least. So its rules are honored by everyone, at least nominally. Let's see...
No browser passes test262 ECMAScript compliance tests completely (WP#Conformance tests). However, no test262 errors found on google are float-related.
IE5.5+ ([MS-ES3]) reports discrepancies in
Number.toFixed
,Number.toExponential
,Number.toPrecision
. Other differences aren't float-related. I couldn't run test262 in IE8;To validate a system, you can use float-related tests from test262. They are located at
http://test262.ecmascript.org/json/ch<2-digit # of spec chapter>.json
; test code can be extracted with (python 2.6+):Another opportunity is IEEE 754 compliance tests in C.
Relevant sections from test262 (ones that compare floating point numbers) are as follows:
this list and the concatenated source of all the relevant test files (as of 3/9/2012, no files from the harness) can be found here: http://pastebin.com/U6nX6sKL
一般经验法则是,当数字精度很重要并且您只能访问浮点精度数字时,所有计算都应以整数数学形式完成,以最好地确保有效性(确保 15 位数据确实有效)。是的,JavaScript 中有很多通用的数字特性,但它们更多地与浮点数缺乏精度相关,而不是与标准的 UA 实现相关。环顾四周,寻找浮点数学的陷阱,它们数量众多且危险。
我觉得我应该详细说明一下,例如我编写了一个程序(用 JavaScript),它使用基本微积分来确定尺寸以米或英尺为单位的多边形的面积。该程序没有按原样进行计算,而是将所有内容转换为微米并在那里进行计算,因为所有内容都会更加完整。
希望这会有所帮助 -ck
为了回应您的澄清、评论和担忧,
我不会在下面完整地重复我的评论,但简短的回答是没有人会这样做永远可以说每个实现都是100% 100% 的设备。时期。我可以说,其他人也会告诉你同样的事情,那就是在当前的主要浏览器上,我没有看到也没有听说过任何涉及浮点数的浏览器特定的有害错误。但你的问题本身就是一把双刃剑,因为你想“依赖”“不可靠”的结果,或者只是你希望所有浏览器“始终不一致”强> - 换句话说,你的时间最好花在寻找一只狗上,而不是试图确保狮子会玩取球游戏,这意味着:你可以110%依赖整数数学以及所述数学的结果,同样如此对于已经向您建议的字符串数学...
祝你好运 -ck
General rule of thumb is that when number precision is important and you only have access to floating point precision numbers, all of your calculations should be done as integer math to best ensure validity (where you're assured 15 digits of assuredly valid data). And yes there are a bunch of general numeric idiosyncrasies in JavaScript but they are more associated with the lack of precision within floating point numbers and not with UA implementations of the standard. Look around for the pitfalls of floating point math, they're numerous and treacherous.
I feel as I should elaborate a little, for instance I wrote a program (in JavaScript) that used basic calculus to determine the area of polygon with dimensions given in meters or feet. Instead of doing the calculations as is, the program converted everything to micrometers and did its calculations there as everything would be more integral.
hope this helps -ck
In response to your clarification, comments and concerns
I'm not going to repeat my comments below in their entirety, however the short answer is no one will ever be able to say that EVERY IMPLEMENTATION is 100% on 100% of devices. Period. I can say and others will tell you the same, is that on the current major browsers I have not seen nor heard of any browser specific detrimental bug involving floating point numbers. But your question itself is kind of a double edged sword since you want to "rely" upon "unreliable" results, or simply that you want all the browsers to be "consistently inconsistent" - in other words instead of trying make sure a lion will play fetch, your time would be better spent looking for a dog, meaning: you can rely 110% on integer math AND the results of said math, the same goes for string math which has already been suggested to you...
good luck -ck
(编辑:下面提到的错误已于 2016 年 3 月 3 日修复并关闭。所以我的答案现在是“也许”。)
不幸的是,答案是否定的。 v8 中至少存在一个突出的错误,由于双舍入,这意味着它可能与 32 位 Linux 上的 IEEE 754 双精度不匹配。
这可以通过以下方式进行测试:
我可以验证这一点在 32 位 Ubuntu 上运行的 Chrome 26.0.1410.63 上失败(左侧为
9007199254740996
)。它在同一系统上传递了 Firefox 20.0。至少,这个测试应该添加到您的测试套件中,也许是 test262。(EDIT: The bug mentioned below was closed as fixed on 3 Mar 2016. So my answer is now "maybe".)
Unfortunately the answer is no. There is at least one outstanding bug in v8 that, due to double-rounding, means it might not match IEEE 754 double precision on 32-bit Linux.
This can be tested with:
I can verify that this fails (the left-hand side is
9007199254740996
) on Chrome 26.0.1410.63 running on 32-bit Ubuntu. It passes on Firefox 20.0 on the same system. At the very least, this test should be added to your test suite, and maybe test262.“我对分布式 Web 应用程序有一个特定的用途,只有当我可以在所有浏览器上获得一致的结果时,它才有效。”
那么答案是否定的。您不能依赖规范来告诉您浏览器正确处理浮动。 Chrome 每 6 周更新一次,因此即使您有规范,Chrome 也可能会在下一版本中更改其行为。
您必须在每次运行计算之前依赖功能测试来测试您的假设。
"I have a specific use in mind for a distributed web application, which would only work if I can rely on consistent results across all browsers."
Then the answer is no. You can not relay on a specification to tell you that a browser correctly handles floats. Chrome updates every 6 weeks, so even if you have the specifications Chrome could change there behavior in the next release.
You have to relay on feature testing that test your assumptions before each time before you calculations is run.
也许您应该使用库进行计算。例如 bignumber 可以很好地处理浮点数。在这里,您应该免受环境更改的影响,因为它使用自己的存储格式。
Maybe you should use a library for your calculations. For example bignumber has a good handling of floating point numbers. Here you should be save from environment changes because it uses it's own storage format.
这是计算领域自古以来的一个问题。如果你问从汇编语言成熟的老程序员,他们会告诉你,你以不同的格式存储重要的数字,并以类似的方式对它们进行操作。
例如,可以通过将浮点值乘以 100(以保持 2 位小数不变)将货币值保存为整数。然后,您可以安全地进行计算,当您必须显示最终结果时,将其除以 100。根据您必须保持安全的小数位数,您可能必须选择 100 以外的其他数字。将内容存储在长期价值,永远不用担心此类问题。
这就是迄今为止在各个平台上给我带来满意结果的原因。我只是这样让自己远离浮点算术的细微差别
This is a problem since ages in computing. And if you ask old programmers who matured from assembly language, they will tell you that you store important numbers in a different format and do manipulations on them in similar way too.
For example, a currency value can be saved as integer by multiplying the float value by 100 (to keep the 2 decimal places intact). You can then safely do calculations and when you have to display the final result, divide there by 100. Depending upon how many decimal places you have to keep secure and safe, you may have to select a different number other than 100. Store things in a long value and be care-free about such problems ever.
This is what gives me satisfactory results across platforms so far. I just keep myself away from the floating point arithmetic nuances this way