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

Mic*_*jko 6 c# linq lambda delegates

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

void Target( params object[] args );
Run Code Online (Sandbox Code Playgroud)

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

Action<int, int> someAction += (a, b) => Target(a, b);
Run Code Online (Sandbox Code Playgroud)

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

someAction += CreateDelegate( typeof(someAction), Target );
Run Code Online (Sandbox Code Playgroud)

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

编辑

将事件重命名为action并将handler更改为目标.

Ree*_*sey 6

这种方法的代表:

void Handle( params object[] args );
Run Code Online (Sandbox Code Playgroud)

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

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

  • 不完全 - "Action <object []>`没有`params`修饰符,因此你不能将它称为`action(foo,bar,baz)`而是*真的*具有该签名的委托*将*以这种方式调用. (3认同)

Mic*_*jko 3

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

Expression<Action<int, int>> ex = (a, b) => Target(a, b);
Run Code Online (Sandbox Code Playgroud)

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

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();
 }
Run Code Online (Sandbox Code Playgroud)