11

我们开发了一种成熟的软件,可以在除一台以外的所有已知计算机上正常工作。问题是解析以“8”开头的字符串。字符串开头的“8”似乎是保留字符。

Parsing:
int.Parse("8") -> Exception message: Input string was not in a correct format. 
int.Parse("80") -> 0
int.Parse("88") -> 8
int.Parse("8100") -> 100

CurrentCulture: sv-SE 
CurrentUICulture: en-US

使用int.Parse("8", CultureInfo.InvariantCulture)解决了这个问题。但是,很高兴知道问题的根源。

问题:如果我们不指定不变的文化,为什么我们会得到这种“8”的行为?


附加信息:

我确实向我的客户发送了一个小程序,实现了上述结果:

    private int ParseInt(string s)
    {
        int parsedInt = -1000;
        try
        {
            parsedInt = int.Parse(s);

            textBoxMessage.Text = "Success: " + parsedInt;

        }
        catch (Exception ex)
        {
            textBoxMessage.Text =
                string.Format("Error parsing string: '{0}'", s) + Environment.NewLine +
                "Exception message: " + ex.Message;
        }

        textBoxMessage.Text += Environment.NewLine + Environment.NewLine +
            "CurrentCulture: " + Thread.CurrentThread.CurrentCulture.Name + "\r\n" +
            "CurrentUICulture: " + Thread.CurrentThread.CurrentUICulture.Name + "\r\n";
        return parsedInt;
    }

更新

我偶然发现了这个链接,这是 microsoft connect 数据库中的一个错误:

https://connect.microsoft.com/VisualStudio/feedback/details/253265/int32-parse-fails-to-convert-the-string-0-zero-on-some-systems

似乎有类似症状的问题,但没有真正的根本原因。如果有人能详细说明这一点,我将不胜感激!

4

1 回答 1

21

因为sv-SE文化8代表CurrencyNegativePattern,这就是为什么你会得到你描述的错误。

您可以通过运行以下示例来检查这一点:

var ci = new CultureInfo("sv-SE");

var nfi = (NumberFormatInfo)ci.GetFormat(typeof(NumberFormatInfo));

Console.WriteLine(nfi.CurrencyNegativePattern);
Console.WriteLine(nfi.CurrencyPositivePattern);

这将输出:

// 8
// 3

Parse通过使用接受NumberStyles枚举的重载,您可以明确地说您正在解析整数而不是货币。

Int32.Parse("8", NumberStyles.Integer, new CultureInfo("sv-SE"));

这一次,由于您指定要解析整数,因此不会发生错误。


但是,IIRCInt32.Parse默认情况下应该将输入解释为整数,所以为什么您会使用该示例代码遇到错误,这超出了我的理解。


更新:

从您最近添加的信息来看,您似乎应该确保问题不是外部问题。也就是说,例如,如果用户将 windows 语言环境的正号设置更改为8正常的,并且对于您获得所获得的错误非常有意义。这就像将 设置+为正号然后尝试解析它:

var ci = new CultureInfo("sv-SE");
var nfi = (NumberFormatInfo)ci.GetFormat(typeof(NumberFormatInfo));

nfi.PositiveSign = "+";

Int32.Parse("+", nfi); // This will throw

向用户询问其区域设置注册表设置,如 Connect 问题中所示,并检查它们是否符合您的期望。

旁注:欢迎使用 SO,顺便说一下,下次您需要向问题添加更多信息时,您应该对其进行编辑,而不是在答案中提供。

于 2010-04-27T09:29:42.760 回答