适当地舍入财务数据

时间:2012-03-15 21:33:47

标签: c# winforms math financial

我决定重新创建我的问题:

decimal dTotal = 0m;
foreach (DictionaryEntry item in _totals)
    {
        if (!string.IsNullOrEmpty(item.Value.ToString()))
        {
            dTotal += Convert.ToDecimal(item.Value);
        }
    }
    Console.WriteLine(dTotal / 3600m);
    Console.WriteLine(decimal.Round(dTotal / 3600m, 2));
    Console.WriteLine(decimal.Divide(dTotal, 3600m));

上面的代码返回:

  

579.99722222222222222222222222

     

580.00

     

579.99722222222222222222222222

所以,这就是我的问题所在,我真的需要它才能显示579.99;但任何一轮,无论是decimal.Round还是Math.Round仍然会返回580;甚至{0:F}的字符串格式也会返回580.00

我该如何正确地做到这一点?

1 个答案:

答案 0 :(得分:10)

新答案(针对新问题)

好的,所以你的值为579.99722222222222222222222222 - 并且要求将其四舍五入到小数点后两位。不是580.00的自然答案吗?它比579.99更接近原始值。这听起来像你本质上想要地板行为,但具有给定的数字位数。为此,您可以使用:

var floored = Math.Floor(original * 100) / 100;

在这种情况下,您可以一步完成两个步骤:

var hours = Math.Floor(dTotal / 36) / 100;

......相当于

var hours = Math.Floor((dTotal / 3600) * 100) / 100;

原始答案(原始问题)

听起来你可能以不恰当的形式开始payTotal

using System;

class Test
{
    static void Main()
    {
        decimal pay = 2087975.7m;
        decimal time = pay / 3600;
        Console.WriteLine(time); // Prints 579.99325
    }
}

这是问题所在:

var payTotal = 2087975.7;

这是将payTotal分配给double变量。你实际获得的价值是2087975.69999999995343387126922607421875,这不是你想要的。每当你发现自己从double投射到decimal或反之亦然时,你应该担心:你可能在某个地方使用了错误的类型。货币值应该绝对存储在decimal而不是double中(还有其他各种Stack Overflow问题,讨论何时使用哪个)。

有关更多信息,请参阅我关于浮点的两篇文章:

(一旦你得到了正确的结果,格式化它们当然是另一回事,但这不应该太糟糕......)