草庐IT

JavaScript数值计算时精度问题处理

Data_Adventure 2023-03-28 原文

js精度问题

当使用 JavaScript 进行数值计算时,会面临一些精度问题,这些问题可能会导致不正确的结果。以下是一些常见的奇奇怪怪的 js 数据精度问题:

1. 浮点数精度问题

在 JS 中,浮点数的精度有限。例如:

0.1 + 0.2 // 结果为0.30000000000000004 23.327*100 // 结果为2332.7000000000003 这个结果显然不符合我们的期望。因为这是由于浮点数本身就无法表示 0.1 和 0.2 精确值。

解决方案:可以使用 toFixed 方法将其转换为字符串保留特定位数的小数,或者使用 Big.js 库中的 Big 对象。

2. 整数运算超出范围

在 JS 中,整数运算的范围为 -2^53 ~ 2^53。当运算结果超出该范围时,会发生奇怪的事情,例如:

Math.pow(2, 53) + 1 // 结果为9007199254740992 解决方案:可以使用 BigInt 类型进行更大范围的整数运算,但是要注意支持性不够广泛,需在浏览器和 Node.js 环境中提供额外的支持。

3. 数字类型转换问题

在 JS 中,数字类型之间的转换也可能会导致奇怪的问题。例如:

"10" > "9" // 结果为false 原因是在字符串比较中,比较的是字符编码而不是实际的数字大小。

解决方案:可以使用 parseInt 或者 parseFloat 将字符串转换为数字,或者使用 Number 函数将字符串强制类型转换为数字。

4. 小数点后多余的零

在 JS 中,将一个小数转换成字符串时,如果它没有小数部分,那么默认会添加 .0。例如:

String(1.0) // 结果为"1.0" 这种自动添加小数点可能会导致输出结果与预期不符。

解决方案:可以使用 Number.prototype.toFixed() 来保持一定的小数点位数,或者使用 Number.prototype.toPrecision() 来控制总位数。

Big.js

Big.js 是一个 JavaScript 的 “任意精度” 数字库,能够处理普通数字无法表示的大数字计算。这个库可以非常有用,例如在金融交易、密码学和科学计算中。

安装

Big.js 可以通过多种途径来安装:

  • npm 安装: npm install big.js
  • 下载源码包: https://github.com/MikeMcl/big.js/archive/v5.2.2.tar.gz
  • 在线 CDN 引入: <script src="https://cdn.jsdelivr.net/npm/big.js@5.2.2/big.min.js"></script> (这里使用了 jsDelivr CDN)

使用方法

在代码中引入 big.js:

const Big = require('big.js');

创建一个 Big 对象

const x = new Big(123.4567); const y = Big(987654321.123456789); const z = new Big('123456789012345678901234567890'); 上述代码创建了三个 Big 对象。你可以将任何字符串、数字或其他 Big 对象传递给构造函数。请注意,只有字符串才能正确表示 0.1、0.01 等小数。我们可以用以下方式来打印这些对象的值:

console.log(x.toString()); // "123.4567" console.log(y.toString()); // "987654321.123456789" console.log(z.toString()); // "123456789012345678901234567890"

进行运算

Big.js 支持 +、-、*、/、mod、pow、sqrt 和 abs 操作。下面是代码示例:

const x = Big(123.4567); const y = Big('987654321.123456789'); console.log(x.plus(y).toString()); // "987654444.580156789" console.log(x.minus(y).toString()); // "-987654197.666756789" console.log(x.times(y).toString()); // "121931366283.89509775303" console.log(x.div(y).toString()); // "0.00012468606749151914" console.log(x.mod(y).toString()); // "123.4567" console.log(x.pow(3).toString()); // "18604128.120667185023" console.log(x.sqrt().toString()); // "11.107774091203273684" console.log(x.abs().toString()); // "123.4567"

设置运算精度

默认情况下,Big.js 将结果四舍五入到 20 位数字。但是你可以更改这个精度设置。这里提供了两种方式:

全局配置

在全局范围内,你可以通过 Big.RM 和 Big.DP 来更改默认设置。RP(Rounding Precision)指定了四舍五入的精度(默认为 20),而 DP(Decimal Places)指定了默认保留的小数位数。

Big.RM = 0; // 舍去模式(0 表示四舍五入) Big.DP = 10; // 小数点后保留 10 位

局部配置

如果你不想全局更改设置,那么可以在每次操作时单独进行设置。以下是如何使用它们:

const x = new Big(1); const y = new Big(3); x.div(y); // '0.33333333333333333333' x.dp = 2; // 将小数点后的位数设置为 2 x.div(y); // '0.33' x.round(1); // 四舍五入至整数 x.div(y).toString(); // '0.3'

十六进制和二进制格式

当处理加密哈希等特殊数据类型时,十六进制和二进制格式的数字显得尤为重要。Luckily, Big.js 提供了相关方法。以下是相关代码示例:

创建一个 big.js 实例需要传入一个数字或字符串。下面是一个创建 big.js 实例的示例:

const num1 = new Big(123.4567); const num2 = new Big("9876543210123456789");

加法

Big 对象可以使用 plus() 方法来进行加法操作,返回一个新的 Big 对象,不改变原有对象:

const num1 = new Big(1.23); const num2 = new Big(4.56); const result = num1.plus(num2); // 等价于 num1 + num2 console.log(result.toString()); // 5.79

减法

Big 对象可以使用 minus() 方法来进行减法操作,返回一个新的 Big 对象,不改变原有对象:

const num1 = new Big(1.23); const num2 = new Big(4.56); const result = num1.minus(num2); // 等价于 num1 - num2 console.log(result.toString()); // -3.33

乘法

Big 对象可以使用 times() 方法来进行乘法操作,返回一个新的 Big 对象,不改变原有对象:

const num1 = new Big(1.23); const num2 = new Big(4.56); const result = num1.times(num2); // 等价于 num1 * num2 console.log(result.toString()); // 5.6088

除法

Big 对象可以使用 div() 方法来进行除法操作,返回一个新的 Big 对象,不改变原有对象:

const num1 = new Big(1.23); const num2 = new Big(4.56); const result = num1.div(num2); // 等价于 num1 / num2 console.log(result.toString()); // 0.26973684210526315789

幂运算

Big 对象可以使用 pow() 方法来进行幂运算操作,返回一个新的 Big 对象,不改变原有对象:

const num1 = new Big(2); const num2 = new Big(3); const result = num1.pow(num2); // 等价于 num1 ** num2 console.log(result.toString()); // 8

比较

Big 对象可以使用 cmp() 方法来进行比较操作,返回一个负数、零或正数,分别表示小于、等于或大于。

const num1 = new Big(1.23); const num2 = new Big(4.56); console.log(num1.cmp(num2)); // -1 console.log(num2.cmp(num1)); // 1 console.log(num1.cmp(num1)); // 0

四舍五入和保留小数位数

Big 对象可以使用 toFixed() 方法来进行四舍五入和保留小数位数操作:

const num1 = new Big(1.23456789); console.log(num1.toFixed(2)); // 1.23 console.log(num1.toFixed(5)); // 1.23457 console.log(num1.toFixed(10)); // 1.2345678900 在处理大数值时,使用 big.js 这样的高精度库可以避免 JavaScript 中 Number 类型的精度问题。

有关JavaScript数值计算时精度问题处理的更多相关文章

  1. ruby - 在 64 位 Snow Leopard 上使用 rvm、postgres 9.0、ruby 1.9.2-p136 安装 pg gem 时出现问题 - 2

    我想为Heroku构建一个Rails3应用程序。他们使用Postgres作为他们的数据库,所以我通过MacPorts安装了postgres9.0。现在我需要一个postgresgem并且共识是出于性能原因你想要pggem。但是我对我得到的错误感到非常困惑当我尝试在rvm下通过geminstall安装pg时。我已经非常明确地指定了所有postgres目录的位置可以找到但仍然无法完成安装:$envARCHFLAGS='-archx86_64'geminstallpg--\--with-pg-config=/opt/local/var/db/postgresql90/defaultdb/po

  2. ruby - 通过 rvm 升级 ruby​​gems 的问题 - 2

    尝试通过RVM将RubyGems升级到版本1.8.10并出现此错误:$rvmrubygemslatestRemovingoldRubygemsfiles...Installingrubygems-1.8.10forruby-1.9.2-p180...ERROR:Errorrunning'GEM_PATH="/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/ruby-1.9.2-p180@global:/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/rub

  3. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  4. ruby-on-rails - 如果为空或不验证数值,则使属性默认为 0 - 2

    我希望我的UserPrice模型的属性在它们为空或不验证数值时默认为0。这些属性是tax_rate、shipping_cost和price。classCreateUserPrices8,:scale=>2t.decimal:tax_rate,:precision=>8,:scale=>2t.decimal:shipping_cost,:precision=>8,:scale=>2endendend起初,我将所有3列的:default=>0放在表格中,但我不想要这样,因为它已经填充了字段,我想使用占位符。这是我的UserPrice模型:classUserPrice回答before_val

  5. ruby-on-rails - 使用一系列等级计算字母等级 - 2

    这里是Ruby新手。完成一些练习后碰壁了。练习:计算一系列成绩的字母等级创建一个方法get_grade来接受测试分数数组。数组中的每个分数应介于0和100之间,其中100是最大分数。计算平均分并将字母等级作为字符串返回,即“A”、“B”、“C”、“D”、“E”或“F”。我一直返回错误:avg.rb:1:syntaxerror,unexpectedtLBRACK,expecting')'defget_grade([100,90,80])^avg.rb:1:syntaxerror,unexpected')',expecting$end这是我目前所拥有的。我想坚持使用下面的方法或.join,

  6. ruby - 通过 RVM (OSX Mountain Lion) 安装 Ruby 2.0.0-p247 时遇到问题 - 2

    我的最终目标是安装当前版本的RubyonRails。我在OSXMountainLion上运行。到目前为止,这是我的过程:已安装的RVM$\curl-Lhttps://get.rvm.io|bash-sstable检查已知(我假设已批准)安装$rvmlistknown我看到当前的稳定版本可用[ruby-]2.0.0[-p247]输入命令安装$rvminstall2.0.0-p247注意:我也试过这些安装命令$rvminstallruby-2.0.0-p247$rvminstallruby=2.0.0-p247我很快就无处可去了。结果:$rvminstall2.0.0-p247Search

  7. ruby - Fast-stemmer 安装问题 - 2

    由于fast-stemmer的问题,我很难安装我想要的任何ruby​​gem。我把我得到的错误放在下面。Buildingnativeextensions.Thiscouldtakeawhile...ERROR:Errorinstallingfast-stemmer:ERROR:Failedtobuildgemnativeextension./System/Library/Frameworks/Ruby.framework/Versions/2.0/usr/bin/rubyextconf.rbcreatingMakefilemake"DESTDIR="cleanmake"DESTDIR=

  8. ruby - 安装 Ruby 时遇到问题(无法下载资源 "readline--patch") - 2

    当我尝试安装Ruby时遇到此错误。我试过查看this和this但无济于事➜~brewinstallrubyWarning:YouareusingOSX10.12.Wedonotprovidesupportforthispre-releaseversion.Youmayencounterbuildfailuresorotherbreakages.Pleasecreatepull-requestsinsteadoffilingissues.==>Installingdependenciesforruby:readline,libyaml,makedepend==>Installingrub

  9. java - 从 JRuby 调用 Java 类的问题 - 2

    我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www

  10. ruby-on-rails - 简单的 Ruby on Rails 问题——如何将评论附加到用户和文章? - 2

    我意识到这可能是一个非常基本的问题,但我现在已经花了几天时间回过头来解决这个问题,但出于某种原因,Google就是没有帮助我。(我认为部分问题在于我是一个初学者,我不知道该问什么......)我也看过O'Reilly的RubyCookbook和RailsAPI,但我仍然停留在这个问题上.我找到了一些关于多态关系的信息,但它似乎不是我需要的(尽管如果我错了请告诉我)。我正在尝试调整MichaelHartl'stutorial创建一个包含用户、文章和评论的博客应用程序(不使用脚手架)。我希望评论既属于用户又属于文章。我的主要问题是:我不知道如何将当前文章的ID放入评论Controller。

随机推荐