将十进制数字四舍五入到不为零的第一个十进制位置

时间:2018-09-21 09:43:28

标签: c# math decimal rounding

我想将数字缩短为不为0的第一个有效数字。后面的数字应四舍五入。

示例:

0.001 -> 0.001
0.00367 -> 0.004
0.00337 -> 0.003
0.000000564 -> 0.0000006
0.00000432907543029 ->  0.000004

当前我有以下步骤:

if (value < (decimal) 0.01)
{
    value = Math.Round(value, 4);
}

注意:

  • 数字将始终为正
  • 有效位数始终为1
  • 大于0.01的值将始终四舍五入到小数点后两位,因此if <0.01

从上面的示例中可以看到,四舍五入到小数点后四位可能是不够的,并且值可能会有很大差异。

5 个答案:

答案 0 :(得分:17)

我将声明precision变量,并使用迭代将该变量乘以10并将其未命中的原始值乘以precision将添加1。 / p>

然后将precision变量用作Math.Round的第二个参数。

static decimal RoundFirstSignificantDigit(decimal input) {
    int precision = 0;
    var val = input;
    while (Math.Abs(val) < 1)
    {
        val *= 10;
        precision++;
    }
    return Math.Round(input, precision);
}

我将为此功能编写扩展方法。

public static class FloatExtension
{
    public static decimal RoundFirstSignificantDigit(this decimal input)
    {
        int precision = 0;
        var val = input;
        while (Math.Abs(val) < 1)
        {
            val *= 10;
            precision++;
        }
        return Math.Round(input, precision);
    }
}

然后使用

decimal input = 0.00001;
input.RoundFirstSignificantDigit();

c# online

结果

(-0.001m).RoundFirstSignificantDigit()                  -0.001
(-0.00367m).RoundFirstSignificantDigit()                -0.004
(0.000000564m).RoundFirstSignificantDigit()             0.0000006
(0.00000432907543029m).RoundFirstSignificantDigit()     0.000004

答案 1 :(得分:2)

类似的东西吗?

    public decimal SpecialRound(decimal value) 
    {
        int posDot = value.ToString().IndexOf('.'); // Maybe use something about cultural (in Fr it's ",")
        if(posDot == -1)
            return value;

        int posFirstNumber = value.ToString().IndexOfAny(new char[9] {'1', '2', '3', '4', '5', '6', '7', '8', '9'}, posDot);

        return Math.Round(value, posFirstNumber);
    }

答案 2 :(得分:1)

var value = 0.000000564;

int cnt = 0;
bool hitNum = false;
var tempVal = value;
while (!hitNum)
{
    if(tempVal > 1)
    {
        hitNum = true;
    }
    else
    {
        tempVal *= 10;
        cnt++;
    }
}

var newValue = (decimal)Math.Round(value, cnt);

答案 3 :(得分:1)

代码来自R,但算法应该很明显

> x = 0.0004932
> y = log10(x)
> z = ceiling(y)
> a = round(10^(y-z),1)
> finally = a*10^(z)
> finally
[1] 5e-04

以下基本上已经由本杰明K提供。

冒着被标记为完整古怪的风险,我高兴地宣布regexp是您的朋友。将您的数字转换为char字符串,搜索第一个不是“。”的字符的位置。或“ 0”,抓住该位置的char和其后的下一个char,将其转换为数字,四舍五入,然后(因为您小心点),将结果乘以$ 10 ^ {-(在“ ”和第一个数字)} $

答案 4 :(得分:0)

另一种方法

    decimal RoundToFirstNonNullDecimal(decimal value)
    {
        var nullDecimals = value.ToString().Split('.').LastOrDefault()?.TakeWhile(c => c == '0').Count();
        var roundTo = nullDecimals.HasValue && nullDecimals >= 1 ? nullDecimals.Value + 1 : 2;
        return Math.Round(value, roundTo);
    }

结果

        Console.WriteLine(RoundToFirstNonNullDecimal(0.001m));                0.001
        Console.WriteLine(RoundToFirstNonNullDecimal(0.00367m));              0.004
        Console.WriteLine(RoundToFirstNonNullDecimal(0.000000564m));          0.0000006
        Console.WriteLine(RoundToFirstNonNullDecimal(0.00000432907543029m));  0.000004
        Console.WriteLine(RoundToFirstNonNullDecimal(0.12m));                 0.12
        Console.WriteLine(RoundToFirstNonNullDecimal(1.232m));                1.23
        Console.WriteLine(RoundToFirstNonNullDecimal(7));                     7.00