3

想象一个具有以下签名的方法:

public void ExampleMethod(string id, object data, 
                          ref object A, ref object B, ref object C)
{
  //...
}

根据 的值,data需要将 in 的值分配给ABCid。简而言之,如果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;
    }
  }
}
4

1 回答 1

2

您不能按名称访问参数,因为您不能真正对变量/参数使用反射。如果这是 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);

或者如果性能很关键,比如fast-member

var wrapped = ObjectAccessor.Create(obj); 
wrapped[id] = data;
于 2012-02-29T09:47:54.943 回答