微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

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

如何解决c#动态事件订阅和取消订阅

我希望能够让一个对象将其方法之一添加到传递给它的 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 关键字可解决此问题,但无法正常取消订阅。有没有办法使这项工作?也许使用指针?

解决方法

这里的问题来自于您传递了 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();
    }
}

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。