2

我有一个使用大量事件的 C# (.NET 4.6.1) 项目。我很想将所有事件处理程序移至新的 WeakEventManager 模式——以避免无休止地担心取消注册处理程序以避免内存泄漏。

但是,我必须对性能进行大量测试,并且想要一种在两种方法之间轻松切换的方法。到目前为止,我一直在使用条件编译来执行以下操作:

#if WeakEvents
    WeakEventManager<EventSource,EArgs>.AddHandler(source, "TheEvent", handler);
#else
    source.TheEvent += handler;
#endif

这行得通,但它很乱。理想情况下,我想创建一个隐藏这个的类。即创建一个内部可以使用任何一种方法的类。然后我可以更改我的所有源以将处理程序附加到新类并轻松切换(甚至将来转移到一些新方法)。

但是我看不到如何编写该类-因为我无法将事件作为参数传递-并且必须对处理程序/名称进行一些反思,这超出了我的范围。

有没有简单的方法可以做到这一点?

4

1 回答 1

1

这是一种方法:

static class EventHelper {
    public static void Subscribe<TSource, TEventArgs>(TSource source, Expression<Func<TSource, EventHandler<TEventArgs>>> eventRef, EventHandler<TEventArgs> handler) where TEventArgs : EventArgs {
        if (source == null)
            throw new ArgumentNullException(nameof(source));
        var memberExp = eventRef.Body as MemberExpression;
        if (memberExp == null)
            throw new ArgumentException("eventRef should be member access expression");

        var eventName = memberExp.Member.Name;
    #if WeakEvents
        WeakEventManager<TSource, TEventArgs>.AddHandler(source, eventName, handler);            
    #else
        // some reflection here to get to the event
        var ev = source.GetType().GetEvent(eventName);
        if (ev == null)
            throw new ArgumentException($"There is no event with name {eventName} on type {source.GetType()}");
        ev.AddMethod.Invoke(source, new object[] { handler });
    #endif
    }
}

用法很简单:

// second parameter is event reference
EventHelper.Subscribe(source, s => s.TheEvent, Handler);
于 2016-10-30T20:24:53.403 回答