带有小数点分隔符的数字错误地转换为双精度

本文关键字:转换 双精度 错误 数字 小数点 分隔符 | 更新日期: 2023-09-27 18:32:01

使用区域性信息时,我在将数字转换为双精度时遇到了奇怪的行为。
使用荷兰文化转换"3,3"时,处理正确。如果我使用美国区域性转换"3,3",它会返回33.我期待一个错误。看我的例子:

static void Main(string[] args)
{
    CultureInfo cultureDutch = new CultureInfo("nl-NL");
    CultureInfo cultureUS = new CultureInfo("en-US");
    System.Threading.Thread.CurrentThread.CurrentCulture = cultureDutch;
    Console.WriteLine("Input 3,3 --> Expected 3,3");
    Console.WriteLine("Output = " + Convert.ToDouble("3,3", cultureDutch));
    // Actual result --> 3,3
    Console.WriteLine("Input 3,3 --> Expected InvalidCastException");
    Console.WriteLine("Output = " + Convert.ToDouble("3,3", cultureUS));
    // Actual result --> 33
    Console.WriteLine();
    Console.WriteLine();
    System.Threading.Thread.CurrentThread.CurrentCulture = cultureUS;
    Console.WriteLine("Input 3.3 --> Expected InvalidCastException");
    Console.WriteLine("Output = " + Convert.ToDouble("3.3", cultureDutch));
    // Actual result --> 33
    Console.WriteLine("Input 3.3 --> Expected 3.3");
    Console.WriteLine("Output = " + Convert.ToDouble("3.3", cultureUS));
    // Actual result --> 3.3
    Console.ReadLine();
}

处理这个问题的正确方法是什么?我更喜欢小数(或千)分隔符无效时的例外。

带有小数点分隔符的数字错误地转换为双精度

如果你只想解析它,我会使用专用的解析方法,您可以在其中设置 数字样式

以下代码将引发 FormatException

var culture =new CultureInfo("en-US");
var result = double.Parse("3,3", NumberStyles.AllowDecimalPoint, culture);

有关详细信息,请参阅 Double.Parse 方法

加上Boas Enkler的回答,我已经能够解决这个问题。因此,首先,我将根据当前区域性扫描输入中的千个分隔符。最后,我将输入值解析为双精度值。

private static string RemoveThousandSeparator(string input)
    {
        Regex removeThousandSeparatorExpr = new Regex(@"^-?(?:'d+|'d{1,3}(?:'"
                        + <CultureInfo>.NumberGroupSeparator + @"'d{3})+)(?:'"
                        + <CultureInfo>.NumberDecimalSeparator + @"'d+)?$");
        Match match = removeThousandSeparatorExpr.Match(input);
        if (match.Success)
        {
            input = input.Replace(<CultureInfo>.NumberGroupSeparator, "");
        }
        else
        {
            throw new Exception("Invalid input value");
        }
        return input;
    }