C# · 12月 31, 2021

c# – 通过反射为任何事件类型订阅动作

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