使用反射将事件处理程序绑定到任何类型的事件

时间:2015-12-31 12:28:44

标签: c# .net events

我有一些代码需要动态地将事件绑定到事件处理程序:

foreach (string evnt in attribute.Events)
{
    EventInfo ei = control.GetType().GetEvent(evnt);
    if(ei != null)
    {
        ei.AddEventHandler(control, new EventHandler((s, e) =>
        {
            // More awesomeness here...
        }));
    }
}

因此,对于事件列表中的每个字符串,从控件获取事件并绑定处理程序。

问题是并非所有事件都是EventHandler,有些事件可能是KeyEventHanderMouseEventHandler等。

我不想要EventHandler派生类型的大量if / else列表,我只想绑定相同的处理程序,无论它是什么类型的EventHandler

我该怎么做?

2 个答案:

答案 0 :(得分:3)

这是一种方法:

首先创建这个助手类:

public class HandlerHelper<T> where T : EventArgs
{
    private readonly EventHandler m_HandlerToCall;

    public HandlerHelper(EventHandler handler_to_call)
    {
        m_HandlerToCall = handler_to_call;
    }

    public void Handle(object sender, T args)
    {
        m_HandlerToCall.Invoke(sender, args);
    }
}

这个泛型类有一个Handle方法,它将用作许多事件处理程序类型的委托。请注意,此类是通用的。 T将是针对不同事件类型的众多EventArgs派生类之一。

现在让我们假设您定义以下事件处理程序:

var event_handler = new EventHandler((s, args) =>
{
    // More awesomeness here...
});

以下是如何使用辅助类为将调用event_handler的不同事件处理程序类型创建不同的委托:

foreach (var event_name in event_names)
{
    var event_info = control.GetType().GetEvent(event_name);

    var event_handler_type = event_info.EventHandlerType;

    var event_args_type = event_handler_type.GetMethod("Invoke").GetParameters()[1].ParameterType;

    var helper_type = typeof(HandlerHelper<>).MakeGenericType(event_args_type);

    var helper = Activator.CreateInstance(helper_type, event_handler);

    Delegate my_delegate = Delegate.CreateDelegate(event_handler_type, helper, "Handle");

    event_info.AddEventHandler(button, my_delegate);
}

对于每个活动,我们都会获得EventHandlerType EventHandlerMouseEventHandler。{/ p>

然后我们使用反射来获取第二个参数的类型,如EventArgsMouseEventArgs

然后我们根据HandlerHelper<>参数的类型创建EventArgs的实例。例如HandlerHelper<EventArgs>HandlerHelper<MouseEventArgs>

我们将event_handler提供给HandlerHelper的构造函数,以便在调用Handle方法时调用它。

这使得Handle方法的签名成为我们想要的。例如,在HandlerHelper<MouseEventArgs>的情况下,Handle方法的签名是:

void Handle(object sender, MouseEventArgs args)

现在,我们使用Delegate.CreateDelegate根据我们创建的特定Handle对象的HandlerHelper方法创建委托,并将此委托提供给AddEventHandler方法

出于性能原因,您可以根据event_args_type缓存委托(而不是每次创建一个委托)。

答案 1 :(得分:0)

您可以使用动态关键字来附加处理程序。您可以在此处找到更多解释:https://msdn.microsoft.com/en-us/library/ms228976(v=vs.110).aspx