如何注销Lambda回调
本文关键字:Lambda 回调 注销 何注销 | 更新日期: 2023-09-27 17:52:59
如果我创建一个lambda回调,如:
var someInstance = new SomeObject();
someInstance.Finished += (obj, args) =>
{
// Do something
// Deregister here
};
someInstance.DoAction();
我如何注销回调作为实际处理程序代码的一部分?这种模式是理想的,因为我可以确保它尽快被释放。
我看到过类似的问题,但没有一个是直接解决这种类型的例子
比如
var someInstance = new SomeObject();
EventHandler myDelegate = null;
myDelegate = (obj, args) =>
{
// Do something
// Deregister here
someInstance.Finished -= myDelegate;
};
someInstance.Finished += myDelegate;
someInstance.DoAction();
您可以在lambda中使用MethodInfo.GetCurrentMethod
来检索lambda的MethodInfo。
使用MethodInfo,您可以使用Delegate.CreateDelegate
来获得代表lambda的正确类型的委托。
使用委托,您可以取消注册lambda,而无需将函数存储在变量中或使其成为命名方法。
class MyClass
{
public event EventHandler TheEvent;
void TestIt()
{
TheEvent += (sender, eventargs) =>
{
Console.WriteLine("Handled!"); // do something in the handler
// get a delegate representing this anonymous function we are in
var fn = (EventHandler)Delegate.CreateDelegate(
typeof(EventHandler), sender,
(MethodInfo)MethodInfo.GetCurrentMethod());
// unregister this lambda when it is run
TheEvent -= fn;
};
// first time around this will output a line to the console
TheEvent(this, EventArgs.Empty);
// second time around there are no handlers attached and it will throw a NullReferenceException
TheEvent(this, EventArgs.Empty);
}
}
如果您想要取消注册的能力,那么最好将lambda定义为Action,这样您就可以这样写:
someInstance.Finished += MyCustomAction;
和
someInstance.Finished -= MyCustomAction;
如果您想在之后删除事件处理程序,您应该使用命名函数:
someInstance.Finished += HandleFinished;
//...
void HandleFinished(object sender, EventArgs args)
{
someInstance.Finished -= HandleFinished;
}
这似乎可以工作:
EventHandler handler = null;
handler = new EventHandler((sender, e) => SomethingHappened -= handler);
SomethingHappened += handler;
请确保您没有将handler
赋值给任何其他值,因为闭包与该变量绑定。这就是Resharper所抱怨的"访问修改闭包"。