4

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

クラスを作成するメソッド呼び出しは、null許容のdoubleがnullでない場合にのみ、MyDataクラスを作成するために使用する3つのnull許容のdoubleを受け取ります。

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クラスを変更せずにこのメソッドを作成するためのより良い方法はありますか?つまり、名前付きパラメーターを条件付きで渡すことはできますか、それともデフォルトの名前付きパラメーター値を表すインジケーターを渡すことができますか?

4

5 に答える 5

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

nullこのようにして、 sを直接渡すことができ、組み合わせ爆発の影響を受けません。

また、デフォルトを変更することにした場合は、このコードのすべてのユーザーを再コンパイルしなくても機能します。

于 2011-06-12T14:39:28.787 に答える
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)); 
    }
于 2011-06-12T15:36:50.233 に答える
0

たぶんあなたはただ欲しい:

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

于 2011-06-12T14:22:56.887 に答える
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);
于 2011-06-12T15:25:38.143 に答える
0

これを行うには、遅延バインディングが必要です。オープンソースフレームワークImpromptuInterfaceを使用すると、名前付き引数を動的に選択するために必要な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());
}
于 2011-06-15T13:18:45.413 に答える