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入门到精通教程
查看: 672|回复: 0

js中toFixed精度问题的解决办法

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

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-8-30 16:03:30 | 显示全部楼层 |阅读模式

    一:四舍五入并不是真正的四舍五入
    这个问题是在测试阶段我们的测试人员提出来的。一开始我也很吃惊,结果待我在控制台试了一些数据之后,我懵逼了,我一直在用的toFixed方法竟然有问题,我竟然糊涂的用它做了很多事!以下是我在chrome上的结果:

    1.35.toFixed(1) // 1.4 正确
    1.335.toFixed(2) // 1.33  错误
    1.3335.toFixed(3) // 1.333 错误
    1.33335.toFixed(4) // 1.3334 正确
    1.333335.toFixed(5)  // 1.33333 错误
    1.3333335.toFixed(6) // 1.333333 错误
    

    果然有问题,只能网上找资料了,结果又发现同样是上面的一段代码,在IE下又小同大异,以下是IE上的结果:

    1.35.toFixed(1) // 1.4 正确
    1.335.toFixed(2) // 1.34  正确
    1.3335.toFixed(3) // 1.334 正确
    1.33335.toFixed(4) // 1.3334 正确
    1.333335.toFixed(5)  // 1.33334 正确
    1.3333335.toFixed(6) // 1.333334 正确
    

    果然IE才是爸爸。难道是浏览器兼容性问题?兼容性问题难道不应该是出在IE中吗?既然找到问题所在,就好下手。我的办法是把要四舍五入的后一位单独拎出来单独判断。

    let result = number.toString();
    const arr = result.split('.');
    const integer = arr[0];
    const decimal = arr[1];
    result = integer + '.' + decimal.substr(0, n);
    const last = decimal.substr(n, 1);
    

    // 四舍五入,转换为整数再处理,避免浮点数精度的损失

    if (parseInt(last, 10) >= 5) {
        const x = Math.pow(10, n);
        result = ((parseFloat(result) * x) + 1) / x;
        result = result.toFixed(n);
    }
    return result;
    

    自己测了几遍,貌似没什么问题,OK~
    二:计算机二进制编码导致的精度问题
    没过多久,测试又提出来页面报错了~~ 心塞啊,怎么可能报错呢?自己debugger,发现页面中的js进了死循环。很明显问题出在toFixed中回调了toFixed,结果没有走出来,继续debugger,又有了进人的发现。以下是控制台测试:

    console.log(2.115 * 100) // 211.50000000000003
    console.log(2.0115 * 1000) // 2011.4999999999998
    

    能告诉我是在闹哪样?好吧,我猜到了,肯定是计算机的进度问题。既然你一直进入循环,我就手动把你拉出来。

    result = (Math.round((parseFloat(result)) * x) + 1) / x;
    

    强制四舍五入取整,不会进死循环了!
    以下是全部代码:
    // toFixed兼容方法

    Number.prototype.toFixed = function (n) {
        if (n > 20 || n < 0) {
            throw new RangeError('toFixed() digits argument must be between 0 and 20');
        }
        const number = this;
        if (isNaN(number) || number >= Math.pow(10, 21)) {
            return number.toString();
        }
        if (typeof (n) == 'undefined' || n == 0) {
            return (Math.round(number)).toString();
        }
    
        let result = number.toString();
        const arr = result.split('.');
    
        // 整数的情况
        if (arr.length < 2) {
            result += '.';
            for (let i = 0; i < n; i += 1) {
                result += '0';
            }
            return result;
        }
    
        const integer = arr[0];
        const decimal = arr[1];
        if (decimal.length == n) {
            return result;
        }
        if (decimal.length < n) {
            for (let i = 0; i < n - decimal.length; i += 1) {
                result += '0';
            }
            return result;
        }
        result = integer + '.' + decimal.substr(0, n);
        const last = decimal.substr(n, 1);
    
        // 四舍五入,转换为整数再处理,避免浮点数精度的损失
        if (parseInt(last, 10) >= 5) {
            const x = Math.pow(10, n);
            result = (Math.round((parseFloat(result) * x)) + 1) / x;
            result = result.toFixed(n);
        }
    
        return result;
    };
    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

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

    本版积分规则

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

    GMT+8, 2025-1-21 15:32 , Processed in 0.127210 second(s), 27 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

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