【MathJS】入门mathjs最佳实践,快速解决前端在数学计算时出现的精度问题

前沿

前端的小伙伴,在项目开发中,比如常见的订单模块中,都会进行一些基础的数学运算时,不可避免的就会遇到让人头疼的精度问题。为什么会出现精度问题,这里就略过,本篇重点介绍解决精度问题。

上手撸代码

安装

npm i --save-dev mathjs

在项目中引入mathjs

import * as math from 'mathjs'

加一个consol方法,方便打印

const consol = function () {
  for (let i = 0; i < arguments.length; i++) {
    console.log(arguments[i])
  }
}

Add(加法运算)

// 之前
consol(0.1 + 0.2) // 0.30000000000000004
// 之后
const addNumber = math.format(
  math.chain(math.bignumber(0.1))
    .add(math.bignumber(0.2))
    .done()
)
consol(addNumber) // 0.3

Chaining 链式调用

在实际开发中,需要快速上手,要在最短的时间内解决问题。我的思路就是写一个最简单的用例,搞清楚它的输入和输出,从而正确的理解它的作用

consol(math.chain(0.1)) // Chain {value: 0.1}

consol(math.chain(0.1).add(0.2)) // Chain {value: 0.30000000000000004}

/*
* 通过在尾部调用一下任意一种方法,返回Chain的value值
* The Chain has all functions available in the math namespace, and has a number of special functions:
    done()     Finalize the chain and return the chain’s value.
    valueOf()  The same as done(), returns the chain’s value.
    toString() Executes math.format(value) onto the chain’s value, returning a string representation of the value.
*/
consol(math.chain(0.1).add(0.2).done()) // 0.30000000000000004
consol(math.chain(0.1).add(0.2).valueOf()) // 0.30000000000000004
consol(math.chain(0.1).add(0.2).toString()) // 0.30000000000000004

consol(typeof math.chain(0.1).add(0.2).done()) // number
consol(typeof math.chain(0.1).add(0.2).valueOf()) // number
consol(typeof math.chain(0.1).add(0.2).toString()) // string

chain调用,返回的是Chain的实例对象,就需要尾部调用,返回chain的value值,如上。

/*
* Function format
* Format a value of any type into a string.
*/
consol(math.format(math.chain(0.1).add(0.2).done())) // 0.30000000000000004
consol(typeof math.format(math.chain(0.1).add(0.2).done())) // string

看上去,math.format()只是把number类型转成string类型,别慌,后面讲bignumber时,才会真正体会到它的价值。

BigNumbers

划重点:对于具有任意精度的计算,math.js支持BigNumber数据类型,bignumber返回一个Decimal类

/*
* BigNumbers
* 对于具有任意精度的计算,math.js支持BigNumber数据类型
* bignumber返回一个Decimal类
*/
const _a = math.bignumber(0.1) // Decimal {s: 1, e: -1, d: Array(1), constructor: ƒ}
const _b = math.add(math.bignumber(0.1), math.bignumber(0.2)) // Decimal {s: 1, e: -1, d: Array(1), constructor: ƒ}
math.format(_b) // 0.3

math.format(),返回了Decimal类的value值。

总结一下,math的链式调用与rxjs差不多,写法非常灵活,可根据实际场景组合使用。来一个简单的例子:

const addNumber = math.format(
  math.chain(math.bignumber(0.1))
    .add(math.bignumber(0.2))
    .done()
)

const a = math.format(
  math.add(math.bignumber(0.1), math.bignumber(0.2))
)

Subtract(减法运算)

consol(1 - 0.9) // 0.09999999999999998

const subtractNumber = math.format(
  math.chain(1)
    .subtract(math.bignumber(0.9))
    .done()
)
consol(subtractNumber) // 0.1
const subtractNumber_1 = math.format(
  math.subtract(math.bignumber(1), math.bignumber(0.9))
)
consol(subtractNumber_1) // 0.1

Multiply(乘法运算)

consol(11 * 1.2 * 3) // 39.599999999999994

const multiplyMumber = math.format(
  math.multiply(math.bignumber(11), math.bignumber(1.2), math.bignumber(3))
)
consol(multiplyMumber) // 39.6

Divide(除法运算)

consol(1.2 / 3) // 0.39999999999999997

const divideNumber = math.format(
  math.divide(math.bignumber(1.2), math.bignumber(3))
)
consol(divideNumber) // 0.4

const divideNumber_1 = math.format(
  math.chain(math.bignumber(1.2)).divide(math.bignumber(3)).done()
)
consol(divideNumber_1) // 0.4

简单总结一下,简单的加减乘除都可以按照上面的实现,要进行组合计算时,就可以使用链式调用.
还有一些场景需要处理,

1.保留N位小数的问题

Round(四舍五入)

// Round 四舍五入,保留2位小数
consol(math.round(12.0355, 3)) // 12.036
consol(math.round(12.0354, 3)) // 12.035
consol(math.round(12.0354, 2)) // 12.04
consol(math.round(12.100, 2)) // 12.1

最后,再来一个综合实例

consol(
  +math.format(
    math.round(
      math.chain(0.1).add(2, 1).subtract(0.1).multiply(1.2, 33).divide(10000).done(),
      3
    )
  )
)
// 0.011879999999999998 => 0.012

掌握到基础用法之后,我们来手动实现一两个方法

手动实现一个Round

实现非常简单,2行代码,没有任何意外的情况。

// toFixed 即使位数不够,也会用0补充
consol(Number(12.035).toFixed(2)) // 12.04
consol(Number(12.1).toFixed(2)) // 12.10
consol(Number(12).toFixed(2)) // 12.00
// string转number
function formatNumber(number, len = 2) {
  // 四舍五入
  number = Number(number).toFixed(len)
  // 去尾部的0
  return Number(number)
}
consol(formatNumber(12.0355, 3)) // 12.036
consol(formatNumber(12.0354, 3)) // 12.035
consol(formatNumber(12.0354, 2)) // 12.04
consol(formatNumber(12.100, 2)) // 12.1

手动实现一个Add

基本逻辑:把需要计算的数字升级(乘以10的n次幂)成计算机能够精度识别的整数,等计算完毕后在降级(除以10的n次幂)

function add(...numList) {
    const base = 10
    const maxPrecision = numList.reduce((curMaxPrecision, num) => {
        const curPrecision = getNumPrecision(num)
        return curMaxPrecision > curPrecision ? curMaxPrecision : curPrecision
    })
    const enlargeScale = Math.pow(base, maxPrecision)
    const startNum = 0
    const counts = numList.reduce((total, num) => {
        return total + num * enlargeScale
    }, startNum)
    return counts / enlargeScale
}

function getNumPrecision(num) {
    return (num.toString().split('.')[1] || '').length
}

consol(0.1 + 0.2 + 0.3)) // 0.6000000000000001
consol(add(0.1, 0.2, 0.3)) // 0.6

其他解决方案
bignumber.js
BigDecimal.js
long.js

你可能感兴趣的:(mathjs,js,精度问题,javascript)