• JavaScript计算精度问题


    一、原因

     js的Number在内存中存储使用的是一种64位双精度浮点数存储方法。其中,1位用来表示符号,0为正,1为负;11位用来表示指数;52位用来表示尾数。他的表示格式为:s * m * ( 2 ^ e ) (s为符号位,m为尾数,e为指数)

    将十进制的小数转换为二进制的小数,采用“乘二取整法”,即小数部分乘以2,取整数部分,剩下的小数部分继续乘以二,取整数部分,直到取到小数部分为0为止。在线进制转换请戳这里

    举例:0.1 + 0.2 = 0.30000000000000004

    0.1(10) 可转换成二进制:0.0001100110011001100110011001100110011001100110011001101(2) 即:

    e = -4; m = 1.1001100110011001100110011001100110011001100110011010

    0.2(10) 可转换成二进制:0.001100110011001100110011001100110011001100110011001101(2) 即:

    e = -3; m = 1.1001100110011001100110011001100110011001100110011010

    二者相加:

    e = -3; m = 0.1100110011001100110011001100110011001100110011001101(52位) +

    e = -3; m = 1.1001100110011001100110011001100110011001100110011010(52位) 结果是:

    e = -3; m = 10.0110011001100110011001100110011001100110011001100111(52位) 即

    e = -2; m = 1.00110011001100110011001100110011001100110011001100111(53位) 根据round to nearest, tie to even(有近取近,无近取偶)的方法取到近似值:

    e = -2; m = 1.0011001100110011001100110011001100110011001100110100(52位) 可转换成十进制:0.30000000000000004

    二、解决方法

    加法:

    add(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; // 有问题。例如2.3 * 100 = 229.99999999999997
        // return (Math.round(arg1 * m) + Math.round(arg2 * m)) / m; // 方法一:四舍五入的方式
        // return (this.mul(arg1, m) + this.mul(arg2, m)) / m; // 方法二:乘法自己封装了一层
    },

    减法:

    sub(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));
        n = (r1 >= r2) ? r1 : r2;
        return ((arg1 * m - arg2 * m) / m).toFixed(n);
    },

    乘法:

    mul(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);
    },

    除法:

    div(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) { }
        r1 = Number(arg1.toString().replace(".", ""));
        r2 = Number(arg2.toString().replace(".", ""));
        return (r1 / r2) * Math.pow(10, t2 - t1);
    },

    大整数比较大小:

    value.toString().length
  • 相关阅读:
    for of 与 for in的区别2
    jQuery的deferred对象详解
    poj 3128 Leonardo's Notebook(置换的幂)
    解决shell脚本中 echo 怎么写入换行到文件
    【解答】关于内核中没开MMU之前的虚拟地址物理地址转换问题
    iOS中怎样加入自己定义的字体
    Android学习笔记(十四)——在执行时加入碎片(附源代码)
    Threads and Anonymous Classes in JAVA
    与一线Linux嵌入式开发project师的对话
    公司实习经验分享
  • 原文地址:https://www.cnblogs.com/fogmisty/p/16158400.html
Copyright © 2020-2023  润新知