-
14-11-2019 - |
题
想象一个具有以下签名的方法:
public void ExampleMethod(string id, object data,
ref object A, ref object B, ref object C)
{
//...
}
中的值 data
需要分配给 A
, B
, C
或什么都不做,取决于 id
. 。简而言之,如果 id == "A"
然后 A = data;
问题在于该方法的主体是由人输入的,但签名是在运行时生成的。因此,不可能对逻辑进行硬编码,因为在设计时不知道有多少个 ref 参数以及它们的名称。这段代码可以插入任意数量的方法中,每个方法都可能具有不同的签名,并且它必须在每个方法中都能工作。
我知道如何获取当前所在方法的所有参数,但我不知道如何为这些参数之一赋值。我正在寻找类似以下内容的内容:
public void ExampleMethod(string id, object data,
ref object A, ???????, ref object Z)
{
MethodBase method = MethodBase.GetCurrentMethod();
foreach (ParameterInfo parameter in method.GetParameters())
{
if (id == parameter.Name)
{
// Problem: assign data to parameter.
return;
}
}
}
解决方案
您无法按名称访问参数,因为您无法真正使用变量/参数的反射。如果这是 IL,但实际上在 C# 中则不然,您可能有一些选择。我的建议是:更改 API,可能涉及数组或(也许更好)字典。考虑:
public void ExampleMethod(string id, object data,
IDictionary<string,object> args) {
args[id] = data;
}
不确定这是否有帮助......但你想做的事情并不是真正适合反思。另一种选择是动态生成此方法,作为构建过程的一部分,或通过 IL。两者都应该没问题。因此它本质上可以生成(作为 C# 或(在运行时)IL):
public void ExampleMethod(string id, object data,
ref object A, ref object B, ref object C)
{
switch(id) {
case "A": A = data; break;
case "B": B = data; break;
case "C": C = data; break;
default: /* do something */
}
}
另一种方法:键入的对象:说你有:
public void ExampleMethod(string id, object data, SomeType obj) {...}
在哪里 obj
是一个具有“A”、“B”、“C”等属性的对象;那么你想要生成的是:
switch(id) {
case "A": obj.A = data; break;
case "B": obj.B = data; break;
case "C": obj.C = data; break;
default: /* do something */
}
这当然可以通过反思来完成:
var prop = obj.GetType().GetProperty(id);
prop.SetValue(obj, data, null);
或者如果性能很关键,比如 快速会员:
var wrapped = ObjectAccessor.Create(obj);
wrapped[id] = data;
不隶属于 StackOverflow