汇总值时的Javascript舍入问题

时间:2012-08-02 03:36:01

标签: javascript floating-point floating-accuracy

  

可能重复:
  Is JavaScript's Math broken?

用户在前两个文本框中输入值,并且在输入时,Javascript(抱歉,没有jQuery,我还没达到它)用于计算精确的总和以及舍入为2位的总和。

为什么我会出现舍入错误,我该怎么做才能纠正错误?

非常感谢。

嗯.... ParseFloat?错误的数据类型?

如果准确的答案好像是在计算器上添加的话,我希望看到的内容。是否有可以使用的parseDecimal或其他数据类型?

![在此处输入图片说明] [1]

    function SumValues() {
        //debugger;
        var txtSubsContrRbtAmt = document.getElementById("<%=txtSubsContrRbtAmt.ClientID%>");
        var txtDeMinAmt = document.getElementById("<%=txtDeMinAmt.ClientID%>");
        var txtTotRbtAmt = document.getElementById("<%=txtTotRbtAmt.ClientID%>");
        var txtRndRbtAmt = document.getElementById("<%=txtRndRbtAmt.ClientID%>");

        var total = Add(txtSubsContrRbtAmt.value, txtDeMinAmt.value);

        txtTotRbtAmt.value = total;
        txtRndRbtAmt.value = RoundToTwoDecimalPlaces(total);
    }

    function Add() {
        var sum = 0;
        for (var i = 0, j = arguments.length; i < j; i++) {
            var currentValue;
            if (isNumber(arguments[i])) {
                currentValue = parseFloat(arguments[i]);
            }
            else {
                currentValue = 0;
            }

            sum += currentValue;
        }
        return sum;
    }

    function RoundToTwoDecimalPlaces(input) {

        return Math.round(input * 100) / 100
    }

    function IsNumeric(input) {
        return (input - 0) == input && input.length > 0;
    }

    function isNumber(n) {
        return !isNaN(parseFloat(n)) && isFinite(n);
    }

  [1]: http://i.stack.imgur.com/5Otrm.png

更新。我正在评估这样的事情:

function AddWithPrecision(a, b, precision) {
        var x = Math.pow(10, precision || 2);
        return (Math.round(a * x) + Math.round(b * x)) / x;
    }

2 个答案:

答案 0 :(得分:2)

对于在金融领域(或任何处理金钱的软件)编写软件的人来说,有一条黄金法则:永远不要使用花车。因此,大多数处理货币的软件只使用整数,并将十进制数表示为数据结构。

这是一种方法:

(注意:此函数会添加两个看起来像数字的字符串)

(补充说明:没有进行错误检查以帮助澄清。也不处理负数)

function addNumberStrings (a,b) {
    a = a.split('.');
    b = b.split('.');
    var a_decimal = a[1] || '0';
    var b_decimal = b[1] || '0';
    diff = a_decimal.length - b_decimal.length;
    while (diff > 0) {
        b_decimal += '0';
        diff --;
    }
    while (diff < 0) {
        a_decimal += '0';
        diff ++;
    }
    var decimal_position = a_decimal.length;

    a = a[0] + a_decimal;
    b = b[0] + b_decimal;

    var result = (parseInt(a,10)+parseInt(b,10)) + '';

    if (result.length < decimal_position) {
        for (var x=result.length;x<decimal_position;x++) {
            result = '0'+result;
        }
        result = '0.'+result
    }
    else {
        p = result.length-decimal_position;
        result = result.substring(0,p)+'.'+result.substring(p);
    }
    return result;
}

*注意:代码已经简化,其他功能被遗漏为作业。

答案 1 :(得分:0)

为了按照你想要的方式修改你的添加,我建议以某种方式计算每个数字中的小数位This method, for instance然后将最大值传递给toFixed,并修剪任何剩余的零。

function AddTwo(n1, n2) {
    var n3 = parseFloat(n1) + parseFloat(n2);
    var count1 = Decimals(n1, '.');
    var count2 = Decimals(n2, '.');
    var decimals = Math.max(count1, count2);
    var result = n3.toFixed(decimals)
    var resultDecimals = Decimals(result, '.');
    if (resultDecimals > 0) {
        return result.replace(/\.?0*$/,'');
    }
    else {
        return result;
    }
}

// Included for reference - I didn't write this
function Decimals(x, dec_sep)
{
    var tmp=new String();
    tmp=x;
    if (tmp.indexOf(dec_sep)>-1)
        return tmp.length-tmp.indexOf(dec_sep)-1;
    else
        return 0;
} 

这是JSFiddle of that