我有一个结构,它将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类?即我是否可以有条件地传递命名参数,还是可以传递指示符来表示默认的命名参数值?
答案 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());
}