怎么解决javascript数字计算丢失精度问题?


本文摘自PHP中文网,作者青灯夜游,侵删。

js数字计算丢失精度问题解决方案

计算机世界里,数字的计算,所有语言都会丢失精度,所以没有万全之策,但在人力范围内,尽量解决。

网上找了一部分代码,发现是有问题的,比如:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

//加法

Number.prototype.myAdd = function(arg2) {

    var arg1 = this;

    if (isNaN(arg2)) {

        return 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.prototype.mySub = function(arg2) {

    var arg1 = this;

    if (isNaN(arg2)) {

        return 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);

}

//乘法

Number.prototype.myMul = function(arg2) {

    var arg1 = this;

    if (isNaN(arg2)) {

        return 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.prototype.myDiv = function(arg2) {

    var arg1 = this;

    if (isNaN(arg2)) {

        return 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).myMul(pow(10, t2 - t1))

    }

}

在计算一些特殊的数字时,仍然有问题:

比如加法:

1

268.34.myDiv(0.83);//321.7505995203837

所以还要优化

我重新做了一版:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

var operationNumber = function (arg1,arg2,operator) {

    var oper=['+','-','*','/'];

    // 不合法的运算

    if (isNaN(arg1)||isNaN(arg2)||oper.indexOf(operator)<0) {

        return NaN;

    }

    // 除以0

    if (operator==='/'&&Number(arg2)===0) {

        return Infinity;

    }

    // 和0相乘

    if (operator==='*'&&Number(arg2)===0) {

        return 0;

    }

    // 相等两个数字相减

    if ((arg1===arg2||Number(arg1)===Number(arg2))&&operator==='-') {

        return 0;

    }

    var r1, r2, max,_r1,_r2;

    try { r1 = arg1.toString().split(".")[1].length } catch (e) { r1 = 0 }

    try { r2 = arg2.toString().split(".")[1].length } catch (e) { r2 = 0 }

    max = Math.max(r1, r2)

    _r1 = max-r1;

    _r2 = max-r2;

    if (_r1!==0) {

        arg1=arg1+'0'.repeat(_r1)

    }

    if (_r2!==0) {

        arg2=arg2+'0'.repeat(_r2)

    }

    arg1 = Number(arg1.toString().replace('.',''))

    arg2 = Number(arg2.toString().replace('.',''))

    var r3 = operator==='*'?(max*2):(operator==='/'?0:max);

    var newNum = eval(arg1+operator+arg2);

    if (r3!==0) {

        var nStr = newNum.toString();

        nStr = nStr.replace(/^-/,'');

        if (nStr.length<r3+1) {

            nStr = '0'.repeat(r3+1-nStr.length)+nStr;

        }

        nStr = nStr.replace(new RegExp('(\\\d{'+r3+'})$'),'.$1');

        if (newNum<0) {

            nStr = '-'+nStr;

        }

        newNum = nStr*1;

    }

    return newNum;

}

//加法

Number.prototype.myAdd = function(arg2) {

    return operationNumber(this,arg2,'+');

}

//减法

Number.prototype.mySub = function(arg2) {

    return operationNumber(this,arg2,'-');

}

//乘法

Number.prototype.myMul = function(arg2) {

    return operationNumber(this,arg2,'*');

}

// 除法

Number.prototype.myDiv = function(arg2) {

    return operationNumber(this,arg2,'/');

}

如果你发现了bug,评论区及时反馈,我及时跟进修复

推荐教程:《JS教程》

以上就是怎么解决javascript数字计算丢失精度问题?的详细内容,更多文章请关注木庄网络博客

相关阅读 >>

javascript如何删除string里某个字符

使用new操作符实例化一个对象的具体步骤是什么

浏览器内核以及浏览器兼容的问题分析

html5 canvas实现中奖转盘的实例代码

js如何读取和保存文件?方法介绍

css元素选择器的运作原理介绍

javascript自我管理是什么

javascript如何捕获窗口关闭事件

vue.js方法与事件的介绍

html引入外部javascript是什么属性

更多相关阅读请进入《javascript》频道 >>




打赏

取消

感谢您的支持,我会继续努力的!

扫码支持
扫码打赏,您说多少就多少

打开支付宝扫一扫,即可进行扫码打赏哦

分享从这里开始,精彩与您同在

评论

管理员已关闭评论功能...