c#动态事件订阅和取消订阅

时间:2021-02-24 12:16:51

标签: c# dynamic eventhandler

我希望能够让一个对象将其方法之一添加到传递给它的 EventHandler 中,并使该方法能够将自身从 EventHandler 中删除。

public class EventRaiser {
    public event EventHandler event1
    public event EventHandler event2
    public void fire() {
        event1?.Invoke(this, null);
        event2?.Invoke(this, null);
    }
}

public class EventSubscriber {
    EventHandler eh;
    public EventSubscriber(EventHandler eh) {
        this.eh = eh;
        eh += receive;
    }

    public void receive(object obj, EventArgs data) {
        // Do stuff.
        if(condition) eh -= receive;
    }
}

public class MainClass {
    public void Main() {
        EventRaiser er = new EventRaiser();
        EventSubscriber es1 = new EventSubscriber(er.event1);
        EventSubscriber es2 = new EventSubscriber(er.event2);
        er.fire();
    }
}

上面的代码无法编译,因为我什至无法将 er.event1er.event2 传递给 EventSubscriber(“该事件只能出现在 += 的左侧...” )。从 event 中删除 EventHandler 关键字可解决此问题,但无法正常取消订阅。有没有办法使这项工作?也许使用指针?

1 个答案:

答案 0 :(得分:0)

这里的问题来自于您传递了 EventHandler,而不是包含 delegate 的列表本身。基本上是指向您的处理程序的“方法指针列表”。

如您所见,在 event1 的声明中,您有关键字 event,当您将它传递到其他地方时会丢失该关键字。 很遗憾,您无法轻松提取 delegate 的“event 持有人”。

基本上,当您想将处理程序注册到事件时,您需要在编译时引用它,以便能够对它进行 +=-=

您可以执行以下操作:

public class EventRaiser
{
    public delegate void Event1(string args);
    public List<Event1> handlers = new List<Event1>();

    public void register(Event1 handler)
    {
        handlers.Add(handler);
    }

    public void unregister(Event1 handler)
    {
        handlers.Remove(handler);
    }
    
    public void fire()
    {
        handlers.ForEach(handler => handler("myEventArgs"));
    }
}
public class EventSubscriber
{
    Action<Event1> registerAction;
    Action<Event1> unregisterAction;
    public EventSubscriber(Action<Event1> register, Action<Event1> unregister)
    {
        registerAction = register;
        unregisterAction = unregister;
        registerAction(receive);
    }

    public void receive(string args)
    {
        // Do stuff.
        unregisterAction(receive);
    }
}
public class MainClass
{
    public void Main()
    {
        EventRaiser er = new EventRaiser();
        EventSubscriber es1 = new EventSubscriber(er.register, er.unregister);                
        er.fire();
    }
}