问题描述
我希望能够让一个对象将其方法之一添加到传递给它的 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.event1
或 er.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();
}
}