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 相关文章推荐
在textarea文本域中显示HTML代码的方法
Mar 06 Javascript
关于window.pageYOffset和document.documentElement.scrollTop
Apr 05 Javascript
High Performance JavaScript(高性能JavaScript)读书笔记分析
May 05 Javascript
Javascript学习笔记之 对象篇(四) : for in 循环
Jun 24 Javascript
基于zepto.js实现仿手机QQ空间的大图查看组件ImageView.js详解
Mar 05 Javascript
介绍JavaScript中Math.abs()方法的使用
Jun 14 Javascript
由浅入深讲解Javascript继承机制与simple-inheritance源码分析
Dec 13 Javascript
浅析jQuery Ajax请求参数和返回数据的处理
Feb 24 Javascript
浅析JavaScript中的array数组类型系统
Jul 18 Javascript
jQuery用noConflict代替$的实现方法
Apr 12 jQuery
微信小程序异步处理详解
Nov 10 Javascript
解决vue项目使用font-awesome,build后路径的问题
Sep 01 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
晶体管来复再生式二管收音机
2021/03/02 无线电
php UTF8 文件的签名问题
2009/10/30 PHP
Alliance vs Liquid BO3 第二场2.13
2021/03/10 DOTA
JavaScript中也使用$美元符号来代替document.getElementById
2010/06/19 Javascript
js网页侧边随页面滚动广告效果实现
2011/04/14 Javascript
jquery ajax方式直接提交整个表单核心代码
2013/08/15 Javascript
解释&&和||在javascript中的另类用法
2014/07/28 Javascript
jQuery实现大转盘抽奖活动仿QQ音乐代码分享
2015/08/21 Javascript
微信小程序 request接口的封装实例代码
2017/04/26 Javascript
vue-music 使用better-scroll遇到轮播图不能自动轮播问题
2018/12/03 Javascript
使用FormData实现上传多个文件
2018/12/04 Javascript
ES6中的迭代器、Generator函数及Generator函数的异步操作方法
2019/05/12 Javascript
TypeScript之调用栈的实现
2019/12/31 Javascript
vue-router为激活的路由设置样式操作
2020/07/18 Javascript
你不知道的SpringBoot与Vue部署解决方案
2020/11/09 Javascript
[02:25]DOTA2英雄基础教程 熊战士
2014/01/03 DOTA
Python中的二叉树查找算法模块使用指南
2014/07/04 Python
Python的Bottle框架中实现最基本的get和post的方法的教程
2015/04/30 Python
python开发之IDEL(Python GUI)的使用方法图文详解
2015/11/12 Python
Python3.6正式版新特性预览
2016/12/15 Python
python 正确保留多位小数的实例
2018/07/16 Python
python用pip install时安装失败的一系列问题及解决方法
2020/02/24 Python
python自动脚本的pyautogui入门学习
2020/04/01 Python
Python控制台实现交互式环境执行
2020/06/09 Python
Scrapy 配置动态代理IP的实现
2020/09/28 Python
python中字典增加和删除使用方法
2020/09/30 Python
CSS3制作彩色进度条样式的代码示例分享
2016/06/23 HTML / CSS
html5指南-5.使用web storage存储键值对的数据
2013/01/07 HTML / CSS
XD健身器材:Kevlar球、Crossfit健身球
2019/03/26 全球购物
企业出纳岗位职责
2014/03/12 职场文书
校优秀毕业生主要事迹
2014/05/26 职场文书
代领毕业证委托书
2014/08/02 职场文书
单位计划生育责任书
2015/05/09 职场文书
小学语文教师竞聘演讲稿范文
2019/08/09 职场文书
漫画《尖帽子的魔法工坊》宣布动画化
2022/04/06 日漫
Python通过loop.run_in_executor执行同步代码 同步变为异步
2022/04/11 Python