带有小数分隔符的数字错误地转换为 Double
Number with decimal separator incorrectly cast to Double
我在使用区域性信息将数字转换为双精度数时遇到奇怪的行为。
使用荷兰文化转换 "3,3"
时,处理正确。如果我使用美国文化转换 "3,3"
,它会 returns 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();
}
正确的处理方法是什么?当小数点(或千位)分隔符无效时,我更喜欢例外。
如果你只想解析它,我会使用专用的解析方法,你可以在其中设置 Numberstyles
以下代码将抛出 FormatException
var culture =new CultureInfo("en-US");
var result = double.Parse("3,3", NumberStyles.AllowDecimalPoint, culture);
有关详细信息,请参阅 Double.Parse Method
结合 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;
}
我在使用区域性信息将数字转换为双精度数时遇到奇怪的行为。
使用荷兰文化转换 "3,3"
时,处理正确。如果我使用美国文化转换 "3,3"
,它会 returns 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();
}
正确的处理方法是什么?当小数点(或千位)分隔符无效时,我更喜欢例外。
如果你只想解析它,我会使用专用的解析方法,你可以在其中设置 Numberstyles
以下代码将抛出 FormatException
var culture =new CultureInfo("en-US");
var result = double.Parse("3,3", NumberStyles.AllowDecimalPoint, culture);
有关详细信息,请参阅 Double.Parse Method
结合 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;
}