如何注销Lambda回调

本文关键字:Lambda 回调 注销 何注销 | 更新日期: 2023-09-27 17:52:59

如果我创建一个lambda回调,如:

var someInstance = new SomeObject();
someInstance.Finished += (obj, args) =>
{
      // Do something
      // Deregister here
};
someInstance.DoAction();

我如何注销回调作为实际处理程序代码的一部分?这种模式是理想的,因为我可以确保它尽快被释放。

我看到过类似的问题,但没有一个是直接解决这种类型的例子

如何注销Lambda回调

比如

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所抱怨的"访问修改闭包"。