6

假设有一个方法接受可变数量的参数:

void Target( params object[] args );

要将其附加到具有具体参数列表的操作,我们可以创建一个 lambda 表达式:

Action<int, int> someAction += (a, b) => Target(a, b);

是否有可能动态创建此 lambda 表达式以便能够将处理程序附加到任何类型的事件?就像是:

someAction += CreateDelegate( typeof(someAction), Target );

我尝试使用Delegate.CreateDelegate,但它希望目标提供一个带有具体参数列表的方法。我觉得这应该是可能的,Expression.Lambda但现在我没有任何成功。你有想法吗?

编辑

将事件重命名为操作,将处理程序重命名为目标。

4

2 回答 2

6

此方法的委托:

void Handle( params object[] args );

将是Action<object[]>,作为代表不能使用params修饰符。您必须执行编译器所做的事情,并将其他方法映射到对象数组中。

关键字由params编译器处理,因此运行时将使用该方法,就好像它只接受一个普通的对象数组一样。为此,您必须构建适当列表的对象数组,用您的对象填充它,然后将执行此操作的方法附加到您的处理程序。

于 2012-10-22T16:07:58.083 回答
3

我通过分析以下行来查看 lambda 背后的表达式:

Expression<Action<int, int>> ex = (a, b) => Target(a, b);

基于此,我创建了一个自己的委托工厂:

public static Delegate CreateDelegate( Type delegateType, Action<object[]> target )
{
    var sourceParameters = delegateType.GetMethod("Invoke").GetParameters();

    var parameters = sourceParameters.Select( p => Expression.Parameter( p.ParameterType, p.Name ) ).ToArray();

    var castParameters = parameters.Select(p => Expression.TypeAs(p, typeof(object))).ToArray();

    var createArray = Expression.NewArrayInit(typeof(object), castParameters);

    var invokeTarget = Expression.Invoke(Expression.Constant(target), createArray);

    var lambdaExpression = Expression.Lambda( delegateType, invokeTarget, parameters);

    return lambdaExpression.Compile();
 }
于 2012-10-22T23:17:54.333 回答