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更改为目标.
这种方法的代表:
void Handle( params object[] args );
Run Code Online (Sandbox Code Playgroud)
会是Action<object[]>,因为代表不能使用params修饰符.您必须执行编译器所做的操作,并将另一个方法映射到对象数组中.
该params关键字是由编译器处理,因此运行时将使用的方法,就好像它只是需要正常对象阵列.为此,您必须构建适当列表的对象数组,用对象填充它,然后将执行此操作的方法附加到处理程序.
我通过分析以下行查看了 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)