Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 751|回复: 0

js、javascript 0.3-0.2 不等于0.1的解决方案

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-9-3 15:59:50 | 显示全部楼层 |阅读模式

    js浮点精度问题,着急解决的看下面一行就够了:

    前端一般都是((0.3*1000)-(0.2*1000))/1000;
    

    JavaScript 只有一种数字类型 Number,而且在Javascript中所有的数字都是以IEEE-754标准格式表示的。浮点数的精度问题不是JavaScript特有的,因为有些小数以二进制表示位数是无穷的。

    十进制           二进制
        0.1              0.0001 1001 1001 1001 ...
        0.2              0.0011 0011 0011 0011 ...
        0.3              0.0100 1100 1100 1100 ...
        0.4              0.0110 0110 0110 0110 ...
        0.5              0.1
        0.6              0.1001 1001 1001 1001 ...

     

    所以比如 1.1,其程序实际上无法真正的表示 ‘1.1',而只能做到一定程度上的准确,这是无法避免的精度丢失:1.09999999999999999。
    在JavaScript中问题还要复杂些,这里只给一些在Chrome中测试数据:

    console.log(1.0-0.9 == 0.1)    //false
    console.log(1.0-0.8 == 0.2)    //false
    console.log(1.0-0.7 == 0.3)    //false
    console.log(1.0-0.6 == 0.4)    //true
    console.log(1.0-0.5 == 0.5)    //true
    console.log(1.0-0.4 == 0.6)    //true
    console.log(1.0-0.3 == 0.7)    //true
    console.log(1.0-0.2 == 0.8)    //true
    console.log(1.0-0.1 == 0.9)    //true

     

    那如何来避免这类 1.0-0.9 != 0.1 的非bug型问题发生呢?下面给出一种目前用的比较多的解决方案, 在判断浮点运算结果前对计算结果进行精度缩小,因为在精度缩小的过程总会自动四舍五入:

    (1.0-0.9).toFixed(digits)  // toFixed() 精度参数digits须在0与20之间
    console.log(parseFloat((1.0-0.9).toFixed(10)) === 0.1)   //true
    console.log(parseFloat((1.0-0.8).toFixed(10)) === 0.2)    //true
    console.log(parseFloat((1.0-0.7).toFixed(10)) === 0.3)    //true
    console.log(parseFloat((11.0-11.8).toFixed(10)) === -0.8)   //true

     

    写成一个方法:

    //通过isEqual工具方法判断数值是否相等
    function isEqual(number1, number2, digits){
      digits = digits == undefined? 10: digits; // 默认精度为10
      return number1.toFixed(digits) === number2.toFixed(digits);
    }
    console.log(isEqual(1.0-0.7, 0.3));  //true
    //原型扩展方式,更喜欢面向对象的风格
    Number.prototype.isEqual = function(number, digits){
      digits = digits == undefined? 10: digits; // 默认精度为10
      return this.toFixed(digits) === number.toFixed(digits);
    }
    console.log((1.0-0.7).isEqual(0.3)); //true

     

    接下来,再来试试浮点数的运算:

    console.log(1.79+0.12)  //1.9100000000000001
    console.log(2.01-0.12)   //1.8899999999999997
    console.log(1.01*1.3)    //1.3130000000000002
    console.log(0.69/10)     //0.06899999999999999

     

    解决方案:

    //说明:javascript的加法结果会有误差,在两个浮点数相加的时候会比较明显。这个函数返回较为精确的加法结果。
    //调用:accAdd(arg1,arg2)
    //返回值:arg1加上arg2的精确结果
    function accAdd(arg1,arg2){
      var r1,r2,m;
      try{r1=arg1.toString().split(".")[1].length}catch(e){r1=0}
      try{r2=arg2.toString().split(".")[1].length}catch(e){r2=0}
      m=Math.pow(10,Math.max(r1,r2))
      return (arg1*m+arg2*m)/m
    }
    //给Number类型增加一个add方法,调用起来更加方便。
    Number.prototype.add = function (arg){
      return accAdd(arg,this);
    }
     
    //减法函数,用来得到精确的减法结果
    //说明:javascript的加法结果会有误差,在两个浮点数相加的时候会比较明显。这个函数返回较为精确的减法结果。
    //调用:accSub(arg1,arg2)
    //返回值:arg1减去arg2的精确结果
    function accSub(arg1,arg2){
      var r1,r2,m,n;
      try{r1=arg1.toString().split(".")[1].length}catch(e){r1=0}
      try{r2=arg2.toString().split(".")[1].length}catch(e){r2=0}
      m=Math.pow(10,Math.max(r1,r2));
      //last modify by deeka
      //动态控制精度长度
      n=(r1>=r2)?r1:r2;
      return ((arg1*m-arg2*m)/m).toFixed(n);
    }
     
    //除法函数,用来得到精确的除法结果
    //说明:javascript的除法结果会有误差,在两个浮点数相除的时候会比较明显。这个函数返回较为精确的除法结果。
    //调用:accDiv(arg1,arg2)
    //返回值:arg1除以arg2的精确结果
    function accDiv(arg1,arg2){
      var t1=0,t2=0,r1,r2;
      try{t1=arg1.toString().split(".")[1].length}catch(e){}
      try{t2=arg2.toString().split(".")[1].length}catch(e){}
      with(Math){
        r1=Number(arg1.toString().replace(".",""))
        r2=Number(arg2.toString().replace(".",""))
        return (r1/r2)*pow(10,t2-t1);
      }
    }
    //给Number类型增加一个div方法,调用起来更加方便。
    Number.prototype.div = function (arg){
      return accDiv(this, arg);
    }
     
    //乘法函数,用来得到精确的乘法结果
    //说明:javascript的乘法结果会有误差,在两个浮点数相乘的时候会比较明显。这个函数返回较为精确的乘法结果。
    //调用:accMul(arg1,arg2)
    //返回值:arg1乘以arg2的精确结果
    function accMul(arg1,arg2) {
      var m=0,s1=arg1.toString(),s2=arg2.toString();
      try{m+=s1.split(".")[1].length}catch(e){}
      try{m+=s2.split(".")[1].length}catch(e){}
      return  Number(s1.replace(".",""))*Number(s2.replace(".",""))/Math.pow(10,m)
    }
    //给Number类型增加一个mul方法,调用起来更加方便。
    Number.prototype.mul = function (arg){
      return accMul(arg, this);
    }
    <br>//验证一下:
    console.log(accAdd(1.79, 0.12));  //1.91
    console.log(accSub(2.01, 0.12));  //1.89
    console.log(accDiv(0.69, 10));    //0.069<br>console.log(accMul(1.01, 1.3));   //1.313

     

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-12-22 12:59 , Processed in 0.056943 second(s), 30 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表