c# - 为同一个匿名方法创建两个委托(delegate)实例不相等

标签 c# .net delegates anonymous-methods equals

考虑以下示例代码:

static void Main(string[] args)
{
   bool same = CreateDelegate(1) == CreateDelegate(1);
}

private static Action CreateDelegate(int x)
{
   return delegate { int z = x; };
}

您会想象这两个委托(delegate)实例比较起来是相等的,就像它们在使用良好的旧命名方法方法 (new Action(MyMethod)) 时一样。它们比较起来并不相等,因为 .NET Framework 为每个委托(delegate)实例提供了一个隐藏的闭包实例。由于这两个委托(delegate)实例各自将其 Target 属性设置为其各自的隐藏实例,因此它们不进行比较。一种可能的解决方案是为匿名方法生成的 IL 将当前实例(this 指针)存储在委托(delegate)的目标中。这将使委托(delegate)能够正确比较,并且从调试器的角度来看也有帮助,因为您会看到您的类是目标,而不是隐藏类。

您可以在我提交给 Microsoft 的错误中阅读有关此问题的更多信息。错误报告还举例说明了我们为什么要使用此功能,以及为什么我们认为应该更改它。如果您觉得这也是一个问题,请通过提供评级和验证来帮助支持它。

https://connect.microsoft.com/VisualStudio/feedback/ViewFeedback.aspx?FeedbackID=489518

您能看出不应更改功能的任何可能原因吗?您认为这是解决问题的最佳行动方案,还是建议我采取不同的途径?

最佳答案

我不太倾向于认为这是一个“错误”。此外,您似乎假设了 CLR 中根本不存在的某些行为。

此处需要了解的重要一点是,每次调用 CreateDelegate 方法时,您都会返回一个新的匿名方法(并初始化一个新的闭包类)。看来您正在使用 delegate 关键字在内部为匿名方法使用某种池。 CLR当然不会这样做。每次调用该方法时,都会在内存中创建匿名方法(与 lambda 表达式一样)的委托(delegate),并且由于等于运算符当然会在这种情况下比较引用,因此这是预期的结果返回 false

虽然您建议的行为在某些情况下可能有一些好处,但实现起来可能会非常复杂,并且更有可能导致不可预测的情况。我认为当前在每次调用时生成新的匿名方法和委托(delegate)的行为是正确的,我怀疑这也是您将在 Microsoft Connect 上获得的反馈。

如果您非常坚持在问题中描述的行为,总有一个选项 memoizing您的 CreateDelegate 函数,这将确保每次为相同的参数返回相同的委托(delegate)。事实上,因为这很容易实现,这可能是微软不考虑在 CLR 中实现它的几个原因之一。

关于c# - 为同一个匿名方法创建两个委托(delegate)实例不相等,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1422928/

相关文章:

c# - 如何通过依赖注入(inject)创建和配置同一类的多个实例?

c# - .Net 接口(interface)的二进制兼容性

c# - 使用 .NET CORE 3.1 进行 SFTP 下载

c++ - Qt 自定义委托(delegate)

ios - 如何使用委托(delegate)从一个 Controller 向另一个 Controller 发送消息?

c# - .NET Framework 中是否存在用于比较的现有委托(delegate)?

c# - 有没有办法将 C# 序列化对象读入 Python?

c# - ASP.Net MVC 3 Unity IoC 处理

.net - IsAjaxRequest 在具有正确 header 的 AngularJs $http.post 上返回 false

.net - global.asax 中的错误无法执行 server.transfer