考虑:
- 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);
- }