我们假设有一个方法可以获取可变数量的参数:
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添加到目标。
答案 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();
}