JavaScript解决浮点数计算不准确问题的方法分析


Posted in Javascript onJuly 09, 2018

本文实例讲述了JavaScript解决浮点数计算不准确问题的方法。分享给大家供大家参考,具体如下:

最近在学习electron框架,想利用这个框架做一个简单的计算器demo。当我对小数进行运算时,发现了一个问题。

0.1+0.2=?

输出结果是:0.30000000000000004

为什么会这样呢?

其实对于浮点数的四则运算,几乎所有的编程语言都会有类似精度误差的问题,只不过在 C++/C#/Java 这些语言中已经封装好了方法来避免精度的问题,而 JavaScript 是一门弱类型的语言,从设计思想上就没有对浮点数有个严格的数据类型,所以精度误差的问题就显得格外突出。

首先我们分析一下为什么会出现这个精度误差?

首先,我们要站在计算机的角度思考 0.1 + 0.2 这个看似小儿科的问题。我们知道,能被计算机读懂的是二进制,而不是十进制,所以我们先把 0.1 和 0.2 转换成二进制看看:

0.1 => 0.0001 1001 1001 1001..(无限循环)
0.2 => 0.0011 0011 0011 0011…(无限循环)

上面我们发现0.1和0.2转化为二进制之后,变成了一个无限循环的数字,这在现实生活中,无限循环我们可以理解,但计算机是不允许无限循环的,对于无限循环的小数,计算机会进行舍入处理。进行双精度浮点数的小数部分最多支持52位,所以两者相加之后得到这么一串 0.0100110011001100110011001100110011001100110011001100 因浮点数小数位的限制而截断的二进制数字,这时候,我们再把它转换为十进制,就成了 0.30000000000000004。

知道了浮点数产生的原因,那么如何处理这个问题呢?

方法1:通过toFixed(num)方法来保留小数。因为这个方法是根据四舍五入来保留小数的,所以最后的计算结果不精确。

方法2:把要计算的数字升级(乘以10的n次幂)成计算机能够精确识别的整数,计算完以后再降级,推荐使用这一种方法。具体代码如下(主要有3个方法):

/*判断obj是否为一个整数*/
function isInteger(obj){
  return Math.floor(obj) === obj;
}
/**
* 将一个浮点数转换成整数,返回整数和倍数
* 如 3.14 》》314 倍数是100
*
*/
function toInteger(floatNum){
  var ret = {times:1,num:0};
  //是整数
  if(isInteger(floatNum)){
    ret.num = floatNum;
    return ret;
  }
  var strfi = floatNum + '';
  //查找小数点的下标
  var dotPos = strfi.indexOf('.');
  console.log('dotPos===='+dotPos);
  //获取小数的位数
  var len = strfi.substr(dotPos+1).length;
  console.log('len===='+len);
  //Math.pow(10,len)指定10的len次幂。
  var time = Math.pow(10,len);
  //将浮点数转化为整数
  var intNum = parseInt(floatNum*time + 0.5,10);
  console.log('intNum===='+intNum);
  ret.times = time;
  ret.num = intNum;
  return ret;
}
/**
*进行运算
*三个参数分别是要运算的两个数和运算符
*/
function operation(a,b,op){
  var o1 = toInteger(a);
  var o2 = toInteger(b);
  var n1 = o1.num;
  var n2 = o2.num;
  var t1 = o1.times;
  var t2 = o2.times;
  var max = t1 > t2 ? t1 : t2;
  var result = null;
  switch(op){
    case 'add':
      if(t1 === t2){
        result = n1 + n2;
      }else if(t1 > t2){
        result = n1 + n2 * (t1/t2);
      }else{
        result = n1 * (t2/t1) + n2;
      }
      return result / max;
      break;
    case 'subtract':
      if(t1 === t2){
        result = n1 - n2;
      }else if(t1 > t2){
        result = n1 - n2 * (t1/t2);
      }else{
        result = n1 * (t2/t1) - n2;
      }
      return result / max;
      break;
    case 'multiply':
      result = (n1 * n2)/(t1 * t2);
      return result;
      break;
    case 'divide':
      result = (n1 / n2)/(t2 / t1);
      return result;
      break;
  }
}

希望本文所述对大家JavaScript程序设计有所帮助。

Javascript 相关文章推荐
自己的js工具 Cookie 封装
Aug 21 Javascript
JavaScript中Boolean对象的属性解析
Oct 21 Javascript
在AngularJS框架中处理数据建模的方式解析
Mar 05 Javascript
js实现String.Fomat的实例代码
Sep 02 Javascript
AngularJS表格样式简单设置方法示例
Mar 03 Javascript
JavaScript中arguments和this对象用法分析
Aug 08 Javascript
vue.js配合$.post从后台获取数据简单demo分享
Aug 11 Javascript
微信小程序实现圆形进度条动画
Nov 18 Javascript
a标签调用js的方法总结
Sep 05 Javascript
vue登录以及权限验证相关的实现
Oct 25 Javascript
Element Carousel 走马灯的具体实现
Jul 26 Javascript
JavaScript语法约定和程序调试原理解析
Nov 03 Javascript
Vue自定义指令封装节流函数的方法示例
Jul 09 #Javascript
JavaScript实现创建自定义对象的常用方式总结
Jul 09 #Javascript
vue-cli配置环境变量的方法
Jul 09 #Javascript
JS逻辑运算符短路操作实例分析
Jul 09 #Javascript
微信小程序中时间戳和日期的相互转换问题
Jul 09 #Javascript
使用async await 封装 axios的方法
Jul 09 #Javascript
bootstrap 弹出框modal添加垂直方向滚轴效果
Jul 09 #Javascript
You might like
PHP5 安装方法
2006/10/09 PHP
php用数组返回无限分类的列表数据的代码
2010/08/08 PHP
php环境套包 dedeampz 伪静态设置示例
2014/03/26 PHP
PHP5.3与5.5废弃与过期函数整理汇总
2014/07/10 PHP
php自定义hash函数实例
2015/05/05 PHP
PHP中使用curl伪造IP的简单方法
2015/08/07 PHP
PHP实现的用户注册表单验证功能简单示例
2019/02/25 PHP
菜鸟javascript基础资料整理2
2010/12/06 Javascript
formvalidator验证插件中有关ajax验证问题
2013/01/04 Javascript
JS+CSS制作DIV层可(最小化/拖拽/排序)功能实现代码
2013/02/25 Javascript
jquery dialog open后,服务器端控件失效的快速解决方法
2013/12/19 Javascript
javascript为下拉列表动态添加数据项
2014/05/23 Javascript
深入理解JavaScript系列(45):代码复用模式(避免篇)详解
2015/03/04 Javascript
javascript作用域问题实例分析
2015/07/13 Javascript
jQuery中get方法用法分析
2016/12/07 Javascript
EasyUI学习之DataGird分页显示数据
2016/12/29 Javascript
js轮播图透明度切换(带上下页和底部圆点切换)
2017/04/27 Javascript
vue自定义一个v-model的实现代码
2018/06/21 Javascript
微信小程序 点击切换样式scroll-view实现代码实例
2019/10/11 Javascript
微信小程序以ssm做后台开发的实现示例
2020/04/08 Javascript
Element DateTimePicker日期时间选择器的使用示例
2020/07/27 Javascript
[13:16]INFAMOUS vs VGJ T BO3
2018/06/07 DOTA
Python的gevent框架的入门教程
2015/04/29 Python
Python面向对象class类属性及子类用法分析
2018/02/02 Python
Python之使用adb shell命令启动应用的方法详解
2019/01/07 Python
Python os模块常用方法和属性总结
2020/02/20 Python
python 利用zmail库发送邮件
2020/09/11 Python
HTML5 拖放功能实现代码
2016/07/14 HTML / CSS
美国内衣第一品牌:Hanes(恒适)
2016/07/29 全球购物
京东港澳售:京东直邮港澳台
2018/01/31 全球购物
专科毕业生求职简历的自我评价
2013/10/12 职场文书
医师定期考核实施方案
2014/05/07 职场文书
党支部对照检查材料
2014/08/25 职场文书
《圆明园的毁灭》教学反思
2016/02/16 职场文书
教您怎么制定西餐厅运营方案 ?
2019/07/05 职场文书
导游词之湖州-太湖
2019/10/11 职场文书