如何在C#中使用可选参数?

Kal*_*lid 459 c# optional-parameters

注意:当C#尚未支持可选参数时(即在C#4之前),就会询问此问题.

我们正在构建一个通过C#类以编程方式生成的Web API.该类有方法GetFooBar(int a, int b),API有一个GetFooBar接受查询参数的方法&a=foo &b=bar.

这些类需要支持可选参数,C#语言不支持这些参数.什么是最好的方法?

Ale*_*lex 1004

惊讶没有人提到C#4.0可选参数的工作原理如下:

public void SomeMethod(int a, int b = 0)
{
   //some code
}
Run Code Online (Sandbox Code Playgroud)

编辑:我知道在问到这个问题的时候,C#4.0并不存在.但是这个问题在Google中仍然排在"C#可选参数"的第一位,所以我想 - 这个答案值得在这里.抱歉.

  • 是的,忽略了这一点,抱歉.但是这个问题在Google中仍然排在"C#可选参数"的第一位,所以无论如何 - 这个答案值得在这里:) (85认同)
  • 在提出问题时,C#4.0不存在. (53认同)
  • 擅长提供最新信息.理想情况下,原始答案将使用当前信息(如C#4.0)进行更新.我相信那些SO最初的想法是一种维基心态,但是每个人都有点害怕编辑其他人的答案. (42认同)
  • 值得注意的是,可选参数必须放在所有非可选参数之后。即你不能有 public void SomeMethod(int b = 0, int a) (9认同)

Tim*_*vis 125

另一种选择是使用params关键字

public void DoSomething(params object[] theObjects)
{
  foreach(object o in theObjects)
  {
    // Something with the Objects…
  }
}
Run Code Online (Sandbox Code Playgroud)

叫做......

DoSomething(this, that, theOther);
Run Code Online (Sandbox Code Playgroud)

  • 这个答案解释了params关键字beautif 10行,MSDN仍然没有在30 + 1做 (56认同)

ste*_*yer 76

在C#中,我通常会使用多种形式的方法:

void GetFooBar(int a) { int defaultBValue;  GetFooBar(a, defaultBValue); }
void GetFooBar(int a, int b)
{
 // whatever here
}
Run Code Online (Sandbox Code Playgroud)

更新: 上面提到的是我用C#2.0做默认值的方式.我现在正在使用的项目是使用C#4.0,它现在直接支持可选参数.这是我在我自己的代码中使用的一个例子:

public EDIDocument ApplyEDIEnvelop(EDIVanInfo sender, 
                                   EDIVanInfo receiver, 
                                   EDIDocumentInfo info,
                                   EDIDocumentType type 
                                     = new EDIDocumentType(EDIDocTypes.X12_814),
                                   bool Production = false)
{
   // My code is here
}
Run Code Online (Sandbox Code Playgroud)

  • 我相信你不能在默认参数中调用构造函数,它们需要“编译时”兼容,而不是“运行时”......还是我错了? (3认同)

Kal*_*lid 44

从这个网站:

http://www.tek-tips.com/viewthread.cfm?qid=1500861&page=1

C#允许使用[Optional]属性(来自VB,但在C#中不起作用).所以你可以有这样的方法:

using System.Runtime.InteropServices;
public void Foo(int a, int b, [Optional] int c)
{
  ...
}
Run Code Online (Sandbox Code Playgroud)

在我们的API包装器中,我们检测可选参数(ParameterInfo p.IsOptional)并设置默认值.目标是将参数标记为可选参数,而不是像参数名称中的"可选"那样使用kludges.

  • 但是如何使用Foo函数呢?Foo(1,1); foo(1,1,null)和foo(1,1,Missing.value)都会抛出异常 (4认同)
  • 如果您尝试调用它,会不会给出编译时错误? (2认同)
  • 奇怪,上面的一些答案比这要好得多. (2认同)

Kep*_*boy 25

你可以使用方法重载...

GetFooBar()
GetFooBar(int a)
GetFooBar(int a, int b)

这取决于方法签名,我给出的示例缺少"int b"only方法,因为它将具有与"int a"方法相同的签名.

你可以使用Nullable类型......

GetFooBar(int? a, int? b)

然后,您可以使用a.HasValue检查是否已设置参数.

另一种选择是使用'params'参数.

GetFooBar(params object[] args)

如果你想使用命名参数,则需要创建一个类型来处理它们,尽管我认为对于web应用程序已经有类似的东西了.


kri*_*_io 24

您可以在C#4.0中使用可选参数而无需担心.如果我们有一个像这样的方法:

int MyMetod(int param1, int param2, int param3=10, int param4=20){....}
Run Code Online (Sandbox Code Playgroud)

当你调用方法时,你可以跳过这样的参数:

int variab = MyMethod(param3:50; param1:10);
Run Code Online (Sandbox Code Playgroud)

C#4.0实现了一个名为"命名参数"的功能,您实际上可以按名称传递参数,当然您可以按照您想要的顺序传递参数:)


Hug*_*len 20

Hello Optional World

如果希望运行时提供默认参数值,则必须使用反射来进行调用.不像这个问题的其他建议那么好,但与VB.NET兼容.

using System;
using System.Runtime.InteropServices;
using System.Reflection;

namespace ConsoleApplication1
{
    class Class1
    {
        public static void sayHelloTo(
            [Optional,
            DefaultParameterValue("world")] string whom)
        {
            Console.WriteLine("Hello " + whom);
        }

        [STAThread]
        static void Main(string[] args)
        {
            MethodInfo mi = typeof(Class1).GetMethod("sayHelloTo");
            mi.Invoke(null, new Object[] { Missing.Value });
        }
    }
}
Run Code Online (Sandbox Code Playgroud)


Ste*_*low 14

一种允许您在任何位置省略任何参数的简单方法是利用可空类型,如下所示:

public void PrintValues(int? a = null, int? b = null, float? c = null, string s = "")
{
    if(a.HasValue)
        Console.Write(a);
    else
        Console.Write("-");

    if(b.HasValue)
        Console.Write(b);
    else
        Console.Write("-");

    if(c.HasValue)
        Console.Write(c);
    else
        Console.Write("-");

    if(string.IsNullOrEmpty(s)) // Different check for strings
        Console.Write(s);
    else
        Console.Write("-");
}
Run Code Online (Sandbox Code Playgroud)

字符串已经是可以为空的类型,因此它们不需要.

使用此方法后,以下调用均有效:

PrintValues (1, 2, 2.2f);
PrintValues (1, c: 1.2f);
PrintValues(b:100);
PrintValues (c: 1.2f, s: "hello");
PrintValues();
Run Code Online (Sandbox Code Playgroud)

当您定义一种方法时,您可以通过命名它们来自由设置所需的参数.有关命名参数和可选参数的更多信息,请参阅以下链接:

命名和可选参数(C#编程指南)@ MSDN


Viv*_*vek 9

我同意斯蒂芬巴尔.但由于它是一个Web服务,最终用户只使用一种形式的web方法比使用同一方法的多个版本更容易.我认为在这种情况下,Nullable Types非常适合可选参数.

public void Foo(int a, int b, int? c)
{
  if(c.HasValue)
  {
    // do something with a,b and c
  }
  else
  {
    // do something with a and b only
  }  
}
Run Code Online (Sandbox Code Playgroud)


小智 7

可选参数用于方法.如果你需要一个类的可选参数,你是:

  • 使用c#4.0:在类的构造函数中使用可选参数,这是我更喜欢的解决方案,因为它更接近于使用方法所做的事情,因此更容易记住.这是一个例子:

    class myClass
    {
        public myClass(int myInt = 1, string myString =
                               "wow, this is cool: i can have a default string")
        {
            // do something here if needed
        }
    }
    
    Run Code Online (Sandbox Code Playgroud)
  • 使用c#4.0之前的c#版本:你应该使用构造函数链接(使用:this关键字),其中更简单的构造函数导致"主构造函数".例:

    class myClass
    {
        public myClass()
        {
        // this is the default constructor
        }
    
        public myClass(int myInt)
            : this(myInt, "whatever")
        {
            // do something here if needed
        }
        public myClass(string myString)
            : this(0, myString)
        {
            // do something here if needed
        }
        public myClass(int myInt, string myString)
        {
            // do something here if needed - this is the master constructor
        }
    }
    
    Run Code Online (Sandbox Code Playgroud)