说我是否在听一个事件:
Subject.NewEvent += delegate(object sender, NewEventArgs e)
{
//some code
});
现在如何取消注册此活动?或者只是让内存泄漏?
答案 0 :(得分:42)
为您的匿名代表实例提供一个名称:
EventHandler<NewEventArg> handler = delegate(object sender, NewEventArgs e)
{
//some code
};
Subject.NewEvent += handler;
Subject.NewEvent -= handler;
答案 1 :(得分:24)
如果您需要取消注册活动,我建议您避免使用事件处理程序的匿名委托。
这种情况下,将此方法分配给本地方法更好 - 您可以彻底取消订阅该事件。
答案 2 :(得分:21)
要在第一次调用时删除处理程序:
//SubjectType Subject = ..... already defined if using (2)
EventHandler handler = null;
handler = delegate(object sender, EventArgs e)
{
// (1)
(sender as SubjectType).NewEvent -= handler;
// or
// (2) Subject.NewEvent -= handler;
// do stuff here
};
Subject.NewEvent += handler;
答案 3 :(得分:6)
您可以创建从所有事件侦听器取消注册的方法。这不完全是你想要的,但有时它可能是有帮助的。例如(这确实有效=)):
class Program {
static void Main(string[] args) {
A someClass = new A();
someClass.SomeEvent += delegate(object sender, EventArgs e) {
throw new NotImplementedException();
};
someClass.ClearEventHandlers();
someClass.FireEvent();
Console.WriteLine("No error.");
}
public class A {
public event EventHandler SomeEvent;
public void ClearEventHandlers() {
Delegate[] delegates = SomeEvent.GetInvocationList();
foreach (Delegate delegate in delegates) {
SomeEvent -= (EventHandler) delegate;
}
}
public void FireEvent() {
if (SomeEvent != null) {
SomeEvent(null, null);
}
}
}
}
答案 4 :(得分:2)
您需要匿名函数的名称,然后,只有名称在范围内时才能执行此操作:
var handler = new EventHandler(delegate(object o, EventArgs e)
{
//do something...
};
Subject.NewEvent += handler;
// later on while handler is still in scope...
Subject.NewEvent -= handler;
答案 5 :(得分:0)
您是否需要因泄漏以外的原因取消注册?
关于“或者只是允许内存泄漏”位,当垃圾收集器清理主题时,您的匿名代表也应该被清除,因此不应该有泄漏。
答案 6 :(得分:0)
还有一个问题(我的)在一些(太多)细节中进入了这个问题:Weak event handler model for use with lambdas。