有条件地传递命名参数

时间:2011-06-12 14:11:32

标签: c# c#-4.0 named-parameters

我有一个结构,它将3个命名参数带入构造函数...

public struct MyData
{
    private readonly double _value1;
    private readonly double _value2;
    private readonly double _value3;

    public MyData(
        double value1 = 1.0,
        double value2 = 2.0,
        double value3 = 3.0)
    {
        _value1 = value1;
        _value2 = value2;
        _value3 = value3;
    }
}

创建类的方法调用接收三个可空的双精度数,只有当可空的双精度数不为空时,我才想用它来创建MyData类...

public MyData CreateMyData(double? value1, double? value2, double? value3)
{
    MyData myData;
    if (value1.HasValue)
    {
        if (value2.HasValue)
        {
            if (value3.HasValue)
            {
                myData = new MyData(value1, value2, value3); 
            }
            else
            {
                myData = new MyData(value1, value2); 
            }
        }
        else
        {
            if (value3.HasValue)
            {
                myData = new MyData(value1, value3: value3); 
            }
            else
            {
                myData = new MyData(value1); 
            }
        }
    }
    else
    {
        if (value2.HasValue)
        {
            if (value3.HasValue)
            {
                myData = new MyData(value2: value2, value3: value3); 
            }
            else
            {
                myData = new MyData(value2: value2); 
            }
        }
        else
        {
            if (value3.HasValue)
            {
                myData = new MyData(value3: value3); 
            }
            else
            {
                myData = new MyData(); 
            }
        }
    }
    return myData;
}

有没有更好的方法来编写此方法而不修改MyData类?即我是否可以有条件地传递命名参数,还是可以传递指示符来表示默认的命名参数值?

5 个答案:

答案 0 :(得分:2)

public MyData(
    double? value1 = null,
    double? value2 = null,
    double? value3 = null)
{
    _value1 = value1 ?? 1.0;
    _value2 = value2 ?? 2.0;
    _value3 = value3 ?? 3.0;
}

这样,您可以直接传递null,不会受combinatorial explosion的影响。

此外,如果您决定更改默认值,即使不重新编译此代码的所有用户,也可以使用默认值。

答案 1 :(得分:1)

检查这个

   public  MyData CreateMyData(double? value1, double? value2, double? value3)
    {
        var ss= typeof(MyData).GetConstructor(new Type[]{typeof(double),typeof(double),typeof(double)});
        var parametesr = ss.GetParameters();
        return new MyData(value1 ?? Convert.ToDouble(parametesr[0].DefaultValue), value2 ?? Convert.ToDouble(parametesr[1].DefaultValue), value3 ?? Convert.ToDouble(parametesr[2].DefaultValue)); 
    }

答案 2 :(得分:0)

也许你只想:

return new MyData(value1 ?? 1.0, value2 ?? 2.0, value3 ?? 3.0);

答案 3 :(得分:0)

var t = typeof (MyData);
var c = t.GetConstructor(new Type[] { typeof(double), typeof(double), typeof(double)});
var p = c.GetParameters();
return new MyData(value1 ?? p[0].DefaultValue, value2 ?? p[1].DefaultValue, value3 ?? p[2].DefaultValue);

答案 4 :(得分:0)

你需要后期绑定才能做到这一点。通过开源框架ImpromptuInterface,您可以访问dynamically pick named arguments所需的DLR功能。

using ImpromptuInterface;

...

public MyData CreateMyData(double? value1, double? value2, double? value3)
{
    var arg = InvokeArg.Create;
    var argList = new List<Object>();
    if(value1.HasValue)
        argList.Add(arg("value1",value1));
    if(value2.HasValue)
        argList.Add(arg("value2",value2));
    if(value3.HasValue)
        argList.Add(arg("value3",value3));

    return Impromptu.InvokeConstructor(typeof(MyData), argList.ToArray());
}