C# 4.0 - 如何处理可选字符串参数

问题描述 投票:0回答:7

此代码无效:

private void Foo(string optionalString = string.Empty)
{
   // do foo.
}

但是这段代码是:

private void Foo(string optionalString = "")
{
   // do foo.
}

为什么?因为 string.Empty 是只读字段,而不是常量,并且可选参数的默认值必须是编译时常量。

所以,关于我的问题......(好吧,关心)

这就是我必须做的:

private const string emptyString = "";

private void Foo(string optionalString = emptyString)
{
   // do foo.
   if (!string.IsNullOrEmpty(optionalString))
      // etc
}

你们如何处理可选字符串参数?

为什么他们不能将 String.Empty 设为编译时常量?

c# .net string c#-4.0 optional-parameters
7个回答
15
投票

嗯... string optionalParm = "" 又出了什么问题?为什么这么糟糕?在这种情况下,您真的认为空字符串需要一个符号常量吗?那这个怎么样?

const int Zero = 0;

void SomeMethod(int optional = Zero) { }

你觉得这很愚蠢吗?


8
投票

如果您不喜欢“”值,您可以使用默认值(字符串)。
我玩过它,这是允许的。

private static void foo(string param = default(string)) {
    if (!string.IsNullOrEmpty(param)) // or param != default(string)
        Console.WriteLine(param);
}

5
投票

代码分析警告 1026 表示不要使用可选参数。使用重载方法是更好的风格,如下所示:

private void Foo()
{
   Foo(string.Empty);
}
private void Foo(string optionalString)
{
   // do foo.
   if (!string.IsNullOrEmpty(optionalString))
      // etc
}

2
投票

处理它们的最佳方法是:

private void Foo(string optionalString = "")
{
   // do foo.
}

所以你不能使用String.Empty。每个人都认识“”,但如果我发现

optionalString = nullString
我就不知道该怎么想了。 如果没有别的事,请命名该事物
emptyString
——它不为空!


1
投票

我正在回答这个问题。

Why can they not make String.Empty a compile-time constant?

这里是通过Reflector对mscorlib.dll中的String.cs进行反汇编代码

public static readonly Empty;
static String()
{
    Empty = "";
    WhitespaceChars = new char[] { 
        '\t', '\n', '\v', '\f', '\r', ' ', '\x0085', '\x00a0', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', 
        ' ', ' ', ' ', ' ', '​', '\u2028', '\u2029', ' ', ''
     };
}

所以在windows平台下,string.Empty就是“”。但你知道吗,Martian 在他们的操作系统中对 Empty 和 WhitespaceChars 有不同的定义。


0
投票

如果你愿意玩输,并将 null、"" 和空白字符视为相同,那么你可以默认为

null
。当用户名和密码是可选字段时,由于与数据库的可信连接的可能性,这变得非常方便。您可以更改此逻辑以将字符串重置为
null
,从而修改断言和
if
。重要的是要有一致的约定。

private void RunSql(string serverName, string databaseName, string userName = null, string password = null)
{
    userName = Strip(userName);
    password = Strip(password);

    // The `MsTest` assert - works in both `Debug` and `Release` modes.
    Assert.AreEqual<bool>(
        userName == String.Empty,
        password == String.Empty,
        "User name and password should be either both empty or both non-empty!");
   Assert.IsFalse(String.IsNullOrWhiteSpace(serverName));
   Assert.IsFalse(String.IsNullOrWhiteSpace(databaseName));

   var cmdBuilder = new StringBuilder();
   cmdBuilder.AppendFormat("sqlcmd -E -S {0} -d {1} ", serverName, databaseName);
   if (userName.Length > 0)
   {
       cmdBuilder.AppendFormat("-U {0} -P {1} ", userName, password);
   }

   // Complete the command string.
   // Run the executable.
}

// Cannot think of a good name. Emptify? MakeNullIfEmpty?
private string Strip(string source)
{
    if (String.IsNullOrWhiteSpace(source))
    {
        return String.Empty;
    }

    return source;
}

0
投票
   private void Foo(string optionalString = emptyString)
    {
        if (string.IsNullOrEmpty(optionalString )) 
        { 
            return; 
        }
    }
© www.soinside.com 2019 - 2024. All rights reserved.