我正在尝试列出处理程序是方法引用的事件处理程序。
要删除特定的处理程序,我需要在列表中找到它。
但是如何比较两个方法引用的代码地址?
要删除特定的处理程序,我需要在列表中找到它。
但是如何比较两个方法引用的代码地址?
- type
- TEventHandler = reference to procedure;
- procedure TestProc;
- begin
- end;
- procedure TForm26.FormCreate(Sender: TObject);
- var
- Handlers: TList<TEventHandler>;
- begin
- Handlers := TList<TEventHandler>.create;
- try
- Handlers.Add(TestProc);
- Handlers.Remove(TestProc); { doesn't work }
- Assert(Handlers.Count=0); { fails }
- Assert(Handlers.IndexOf(TestProc)>=0); { fails }
- finally
- FreeAndNil(Handlers);
- end;
- end;
解决方法
这不像看起来那么容易。
要了解为什么会发生这种情况,您需要了解编译器如何执行方法引用的分配。
您编写的代码基本上由编译器翻译成:
- Handlers.Add(procedure begin TestProc; end);
- Handlers.Remove(procedure begin TestProc; end);
现在我们必须知道,如果您在同一个例程中有多个匿名方法,那么即使他们的代码是相同的,它们实际上也是不同的匿名方法。 (见How are anonymous methods implemented under the hood?)
这意味着传递给“添加和删除”的值即使其身体中的代码是相同的,即使是黑客也需要二进制代码分析来确定正文中的代码是否相同。
如果你改变代码如下,它将工作,因为你只有一个匿名方法 – 为这个剪辑它的工作,但通常你不会添加和删除在完全相同的例程:
- var
- Handlers: TList<TEventHandler>;
- Handler: TEventHandler;
- begin
- Handlers := TList<TEventHandler>.create;
- try
- Handler := TestProc;
- Handlers.Add(Handler);
- Handlers.Remove(Handler);
- Assert(Handlers.Count=0);
- finally
- FreeAndNil(Handlers);
- end;
- end;
如果您想要添加和删除事件处理程序的列表,我的个人建议是避免匿名方法类型并使用过程或方法:
- type
- TEventHandlerA = procedure;
- TEventHandlerB = procedure of object;
最好的决定取决于你,因为你的代码更好。