考虑:
someControl.Click += delegate { Foo(); };
事件的论据是无关紧要的,我不需要他们,我对他们不感兴趣.我只想要Foo()来调用.没有明显的方法通过反思来做同样的事情.
我想把上面的内容翻译成一些东西
void Foo() { /* launch missiles etc */ } void Bar(object obj,EventInfo info) { Action callFoo = Foo; info.AddEventHandler(obj,callFoo); }
此外,我不想假设传递给Bar的对象类型严格遵守使用事件的EventHander(TArgs)签名的准则.简单来说,我正在寻找一种方式来订阅Action到任何处理程序类型;不太简单,一种将Action代理转换为预期处理程序类型的委托的方法.
解决方法
static void AddEventHandler(EventInfo eventInfo,object item,Action action) { var parameters = eventInfo.EventHandlerType .GetMethod("Invoke") .GetParameters() .Select(parameter => Expression.Parameter(parameter.ParameterType)) .ToArray(); var handler = Expression.Lambda( eventInfo.EventHandlerType,Expression.Call(Expression.Constant(action),"Invoke",Type.EmptyTypes),parameters ) .Compile(); eventInfo.AddEventHandler(item,handler); } static void AddEventHandler(EventInfo eventInfo,Action<object,EventArgs> action) { var parameters = eventInfo.EventHandlerType .GetMethod("Invoke") .GetParameters() .Select(parameter => Expression.Parameter(parameter.ParameterType)) .ToArray(); var invoke = action.GetType().GetMethod("Invoke"); var handler = Expression.Lambda( eventInfo.EventHandlerType,invoke,parameters[0],parameters[1]),handler); }
用法:
Action action = () => BM_21_Grad.LaunchMissle(); foreach (var eventInfo in form.GetType().GetEvents()) { AddEventHandler(eventInfo,form,action); }