js 加减乘除精度问题

   //加法
    Number.prototype.add = function (arg) {
        var r1, r2, m;
        try { r1 = this.toString().split(".")[1].length } catch (e) { r1 = 0 }
        try { r2 = arg.toString().split(".")[1].length } catch (e) { r2 = 0 }
        m = Math.pow(10, Math.max(r1, r2))
        return (this * m + arg * m) / m
    }
    //减法
    Number.prototype.sub = function (arg) {
        return this.add(-arg);
    }
    //乘法
    Number.prototype.mul = function (arg) {
        var m = 0, s1 = this.toString(), s2 = arg.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.prototype.div = function (arg) {
        var t1 = 0, t2 = 0, r1, r2;
        try { t1 = this.toString().split(".")[1].length } catch (e) { }
        try { t2 = arg.toString().split(".")[1].length } catch (e) { }
        with (Math) {
            r1 = Number(this.toString().replace(".", ""))
            r2 = Number(arg.toString().replace(".", ""))
            return (r1 / r2) * pow(10, t2 - t1);
        }
    }


    var floatObj = function () {

        /*
         * 判断obj是否为一个整数
         */
        function isInteger(obj) {
            return Math.floor(obj) === obj
        }

        /*
         * 将一个浮点数转成整数,返回整数和倍数。如 3.14 >> 314,倍数是 100
         * @param floatNum {number} 小数
         * @return {object}
         *   {times:100, num: 314}
         */
        function toInteger(floatNum) {
            var ret = { times: 1, num: 0 }
            if (isInteger(floatNum)) {
                ret.num = floatNum
                return ret
            }
            var strfi = floatNum + ''
            var dotPos = strfi.indexOf('.')
            var len = strfi.substr(dotPos + 1).length
            var times = Math.pow(10, len)
            var intNum = parseInt(floatNum * times + 0.5, 10)
            ret.times = times
            ret.num = intNum
            return ret
        }

        /*
         * 核心方法,实现加减乘除运算,确保不丢失精度
         * 思路:把小数放大为整数(乘),进行算术运算,再缩小为小数(除)
         *
         * @param a {number} 运算数1
         * @param b {number} 运算数2
         * @param digits {number} 精度,保留的小数点数,比如 2, 即保留为两位小数
         * @param op {string} 运算类型,有加减乘除(add/subtract/multiply/divide)
         *
         */
        function operation(a, b, digits, 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) { // o1 小数位 大于 o2
                        result = n1 + n2 * (t1 / t2)
                    } else { // o1 小数位 小于 o2
                        result = n1 * (t2 / t1) + n2
                    }
                    return result / max
                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
                case 'multiply':
                    result = (n1 * n2) / (t1 * t2)
                    return result
                case 'divide':
                    result = (n1 / n2) * (t2 / t1)
                    return result
            }
        }

        // 加减乘除的四个接口
        function add(a, b, digits) {
            return operation(a, b, digits, 'add')
        }
        function subtract(a, b, digits) {
            return operation(a, b, digits, 'subtract')
        }
        function multiply(a, b, digits) {
            return operation(a, b, digits, 'multiply')
        }
        function divide(a, b, digits) {
            return operation(a, b, digits, 'divide')
        }

         function toFixed(num, s) {
            try {
                s = s || (num + '').split(".")[1].length;
            } catch (e) { }
            var times = Math.pow(10, s + 1)
            var des = num * times + 5
            des = parseInt(des / 10, 10) * 10 / times
            return des + ''
        }

        // exports
        return {
            add: add,
            subtract: subtract,
            multiply: multiply,
            divide: divide,
            toFixed: toFixed
        }
    }();

    // toFixed 修复


    const abc = floatObj.toFixed(2.1,2)
 // 加法
const 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
}
// 减法
const subtr = (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)
}
// 乘法获取准确值
const accMul = (arg1, arg2) => {
  var m = 0; var s1 = arg1.toString(); var s2 = arg2.toString()
  try { m += s1.split('.')[1].length } catch (e) { console.log(e) }
  try { m += s2.split('.')[1].length } catch (e) { console.log(e) }
  return Number(s1.replace('.', '')) * Number(s2.replace('.', '')) / Math.pow(10, m)
}
// 除法获取精确值
const accDiv = (arg1, arg2) => {
  var t1 = 0; var t2 = 0; var r1; var r2
  try { t1 = arg1.toString().split('.')[1].length } catch (e) { console.log(e) }
  try { t2 = arg2.toString().split('.')[1].length } catch (e) { console.log(e) }
  r1 = Number(arg1.toString().replace('.', ''))
  r2 = Number(arg2.toString().replace('.', ''))
  return (r1 / r2) * Math.pow(10, t2 - t1)
}

const accRound = (value, n) => {
  if (!n) n = 2
  return Math.round(value * Math.pow(10, n)) / Math.pow(10, n)
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值