ChatGPT解决这个技术问题 Extra ChatGPT

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

注意:这个问题是在 C# 还不支持可选参数的时候提出的(即在 C# 4 之前)。

我们正在构建一个从 C# 类以编程方式生成的 Web API。该类有方法 GetFooBar(int a, int b),API 有一个方法 GetFooBar,采用像 &a=foo &b=bar 这样的查询参数。

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

或者等待 unitil C# 4.0 发布。支持可选参数。

A
Alex from Jitbit

没有人提到像这样工作的 C# 4.0 可选参数感到惊讶:

public void SomeMethod(int a, int b = 0)
{
   //some code
}

编辑:我知道在提出问题时,C# 4.0 并不存在。但是这个问题在“C# 可选参数”中仍然在谷歌中排名第一,所以我想——这个答案值得在这里。对不起。


在提出问题时,C# 4.0 并不存在。
是的,忽略了这一点,对不起。但是这个问题仍然在谷歌中排名第一,因为“C# 可选参数”所以无论如何 - 这个答案值得在这里:)
很高兴您提供最新信息。理想情况下,原始答案将使用 C# 4.0 等当前信息进行更新。我相信这就是 SO 人最初的想法,一种 Wiki 心态,但每个人都有点害怕编辑别人的答案。
我在 WebServices 中发现的一些有趣的事情是(当然在我正在测试的项目中)查询字符串中的所有布尔值默认情况下都是可选的。显然,它们默认为 FALSE。
值得注意的是,可选参数必须放在所有非可选参数之后。即你不能有 public void SomeMethod(int b = 0, int a)
R
Robert Harvey

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

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

被称为...

DoSomething(this, that, theOther);

这个答案在 10 行中很好地解释了 params 关键字,MSDN 仍然无法在 30 中做到。+1
谢谢!这启发我编写了一个简单的(针对我当前的用例)日志记录函数:public void log (params object[] args){ StringBuilder sb = new StringBuilder(); for (int i = 0; i < args.Length; i++){ sb.Append("{"); sb.Append(i.ToString()); sb.Append("}"); sb.Append(" "); } String.Format(sb.ToString(),args).Dump(); } 示例调用:log("...Done,",(watch.ElapsedMilliseconds/1000).ToString(),"s");
你能提一下,net 3.5是否支持它? official docs 没有提到这一点。
@T.Todua - 我不是 100% 确定你在这里问什么 - dotnet 框架 3.5 包括 C# v3.0 - 3.0 版确实支持 params 关键字,但没有可选参数,这是在 C# v4.0 中引入的, 包含在 dotnet framework 4.0 中
s
stephenbayer

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

void GetFooBar(int a) { int defaultBValue;  GetFooBar(a, defaultBValue); }
void GetFooBar(int a, int b)
{
 // whatever here
}

更新:上面提到的这就是我使用 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
}

我相信你不能在默认参数中调用构造函数,它们需要“编译时”兼容,而不是“运行时”......或者我错了吗?
K
Kalid

从这个网站:

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)
{
  ...
}

在我们的 API 包装器中,我们检测可选参数 (ParameterInfo p.IsOptional) 并设置默认值。目标是将参数标记为可选,而无需在参数名称中使用“可选”之类的杂乱无章的东西。


但是如何使用函数 Foo 呢? Foo(1,1);foo(1,1,null) 和 foo(1,1,Missing.value) 都会抛出异常
奇怪,上面的一些答案比这要好得多。
P
Palec

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

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

这取决于方法签名,我给出的示例缺少“int b”唯一方法,因为它与“int a”方法具有相同的签名。

你可以使用 Nullable 类型...

GetFooBar(int? a, int? b)

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

另一种选择是使用“params”参数。

GetFooBar(params object[] args)

如果您想使用命名参数,则需要创建一种类型来处理它们,尽管我认为 Web 应用程序已经有了类似的东西。


k
kristi_io

您可以毫无顾虑地使用 C# 4.0 中的可选参数。如果我们有这样的方法:

int MyMetod(int param1, int param2, int param3=10, int param4=20){....}

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

int variab = MyMethod(param3:50; param1:10);

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


真正在方法调用中的参数之间使用分号-> MyMethod(param3:50; param1:10); 我认为param3:50 之后必须有一个逗号 -> MyMethod(param3:50, param1:10);
S
SteakOverflow

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

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("-");
}

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

一旦你有了这个方法,下面的调用都是有效的:

PrintValues (1, 2, 2.2f);
PrintValues (1, c: 1.2f);
PrintValues(b:100);
PrintValues (c: 1.2f, s: "hello");
PrintValues();

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

Named and Optional Arguments (C# Programming Guide) @ MSDN


H
Hugh Allen

你好可选世界

如果您希望运行时提供默认参数值,则必须使用反射来进行调用。不如这个问题的其他建议好,但与 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 });
        }
    }
}

V
Vivek

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

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
  }  
}

+1 忠告。不要养成这种习惯,因为它会变得非常混乱。
S
SteveC

可选参数用于方法。如果您需要一个类的可选参数并且您是:

使用 c# 4.0:在类的构造函数中使用可选参数,这是我更喜欢的解决方案,因为它更接近于方法所做的事情,因此更容易记住。这是一个例子: class myClass { public myClass(int myInt = 1, string myString = "哇,这太酷了:我可以有一个默认字符串") { // 如果需要,可以在这里做一些事情 } }

使用 c#4.0 之前的 c# 版本:您应该使用构造函数链接(使用 :this 关键字),其中更简单的构造函数会导致“主构造函数”。示例: class myClass { public myClass() { // 这是默认构造函数 } public myClass(int myInt) : this(myInt, "whatever") { // 如果需要在这里做一些事情 } public myClass(string myString) : this (0, myString) { // 如果需要,在这里做一些事情 } public myClass(int myInt, string myString) { // 如果需要,在这里做一些事情 - 这是主构造函数 } }


c
cfbarbero

如 stephen 所述,在 C# 中处理此问题的典型方法是重载该方法。通过创建具有不同参数的方法的多个版本,您可以有效地创建可选参数。在具有较少参数的表单中,您通常会调用该方法的表单,其中所有参数设置您在调用该方法时的默认值。


佚名

使用重载或使用 C# 4.0 或更高版本

 private void GetVal(string sName, int sRoll)
 {
   if (sRoll > 0)
   {
    // do some work
   }
 }

 private void GetVal(string sName)
 {
    GetVal("testing", 0);
 }

T
T.S.

你可以重载你的方法。一种方法包含一个参数 GetFooBar(int a),另一种方法包含两个参数 GetFooBar(int a, int b)


4
4b0

您可以使用默认值。

public void OptionalParameters(int requerid, int optinal = default){}

P
Pang

对于较大数量的可选参数,Dictionary<string,Object> 的单个参数可以与 ContainsKey 方法一起使用。我喜欢这种方法,因为它允许我单独传递 List<T>T 而无需创建一个完整的其他方法(例如,如果将参数用作过滤器,那就太好了)。

示例(如果不需要可选参数,将传递 new Dictionary<string,Object>()):

public bool Method(string ParamA, Dictionary<string,Object> AddlParams) {
    if(ParamA == "Alpha" && (AddlParams.ContainsKey("foo") || AddlParams.ContainsKey("bar"))) {
        return true;
    } else {
        return false;
    }
}

R
Robert Paulson

而不是默认参数,为什么不从传递的查询字符串构造一个字典类.. 一个几乎与 asp.net 表单处理查询字符串的方式相同的实现。

即 Request.QueryString["a"]

这将使叶子类与工厂/样板代码分离。

您可能还想查看 Web Services with ASP.NET。 Web 服务是通过 C# 类的属性自动生成的 Web api。


R
Ron K

聚会有点晚了,但我一直在寻找这个问题的答案,最终想出了另一种方法来做到这一点。将 Web 方法的可选参数的数据类型声明为 XmlNode 类型。如果可选的 arg 被省略,这将被设置为 null,如果它存在,您可以通过调用 arg.Value 获得字符串值,即

[WebMethod]
public string Foo(string arg1, XmlNode optarg2)
{
    string arg2 = "";
    if (optarg2 != null)
    {
        arg2 = optarg2.Value;
    }
    ... etc
}

这种方法的另一个优点是 .NET 生成的 ws 主页仍然显示参数列表(尽管您确实丢失了方便的文本输入框进行测试)。


这比使用可空类型更好吗?
S
Spanky

我要编写一个需要 7 个参数的 Web 服务。每个都是此 Web 服务包装的 sql 语句的可选查询属性。因此,我想到了两种非可选参数的解决方法......都很差:

方法1(参数1,参数2,参数3,参数4,参数5,参数6,参数7)方法1(参数1,参数2,参数3,参数4,参数5,参数6)方法1(参数1,参数2,参数3,参数4,参数5,参数7) )...开始看图。这种方式是疯狂的。组合太多了。

现在寻找一种看起来很尴尬但应该可以使用的更简单的方法:method1(param1, bool useParam1, param2, bool useParam2, etc...)

这是一个方法调用,所有参数的值都是必需的,它将处理其中的每种情况。从界面上如何使用它也很清楚。

这是一个黑客,但它会工作。


这就是存在可空参数的原因。
M
Matt

我必须在 VB.Net 2.0 Web 服务中执行此操作。我最终将参数指定为字符串,然后将它们转换为我需要的任何内容。使用空字符串指定了一个可选参数。不是最干净的解决方案,但它有效。请注意捕获所有可能发生的异常。


C
CodeArtist

以防万一有人想将回调(或 delegate)作为可选参数传递,可以这样做。

可选回调参数:

public static bool IsOnlyOneElement(this IList lst, Action callbackOnTrue = (Action)((null)), Action callbackOnFalse = (Action)((null)))
{
    var isOnlyOne = lst.Count == 1;
    if (isOnlyOne && callbackOnTrue != null) callbackOnTrue();
    if (!isOnlyOne && callbackOnFalse != null) callbackOnFalse();
    return isOnlyOne;
}

u
user3555836

可选参数不过是默认参数!我建议你给他们两个默认参数。 GetFooBar(int a=0, int b=0) 如果你没有任何重载方法,将导致a=0,如果你不传递任何值,b=0,如果你传递1个值,将导致, 为 a 传递值,0,如果传递 2 个值,第一个将分配给 a,第二个分配给 b。

希望这能回答你的问题。


S
Sharunas Bielskis

在默认值不可用的情况下,添加可选参数的方法是使用 .NET OptionalAttribute 类 - https://docs.microsoft.com/en-us/dotnet/api/system.runtime.interopservices.optionalattribute?view=netframework-4.8

代码示例如下:

namespace OptionalParameterWithOptionalAttribute
{
    class Program
    {
        static void Main(string[] args)
        {
            //Calling the helper method Hello only with required parameters
            Hello("Vardenis", "Pavardenis");
            //Calling the helper method Hello with required and optional parameters
            Hello("Vardenis", "Pavardenis", "Palanga");
        }
        public static void Hello(string firstName, string secondName, 
            [System.Runtime.InteropServices.OptionalAttribute] string  fromCity)
        {
            string result = firstName + " " + secondName;
            if (fromCity != null)
            {
                result += " from " + fromCity;
            }
            Console.WriteLine("Hello " + result);
        }

    }
}

A
Ankit Panwar

你也可以试试这个
Type 1
public void YourMethod(int a=0, int b = 0) { //some code }


类型 2
public void YourMethod(int? a, int? b) { //some code }