如何使用可变数量的参数为目标创建委托

时间:2012-10-22 16:06:34

标签: c# linq delegates lambda

我们假设有一个方法可以获取可变数量的参数:

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,但是现在我没有任何成功。你有想法吗?

修改

将事件重命名为action并将handler添加到目标。

2 个答案:

答案 0 :(得分:6)

此方法的代表:

void Handle( params object[] args );

Action<object[]>,因为代理人无法使用params修饰符。您必须执行编译器所做的操作,并将另一个方法映射到对象数组中。

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

答案 1 :(得分: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();
 }