情况
我们正在运行一个大型 WPF 应用程序,该应用程序在很长一段时间内都不会释放内存。这不是真正的内存泄漏,因为内存最终会被释放。我知道通常情况下,这不会被认为是一个问题。不幸的是,它与 WPF 命令基础架构一起成为性能问题。有关更详细的说明,请参见下文。
发现
我们有执行典型用例的自动化测试。有些情况下工作正常并及时释放内存。其他人正在占用内存,直到客户端最小化、打开新窗口或发生触发 Gen2 收集的其他情况。
• 在ANTS 中,我们看到对象没有GC Root,但有很多对其他需要终结的对象的引用。
• WinDbg 不显示任何准备完成的对象。
• 运行几个GC.Collect()
,GC.WaitForPendingFinalizers()
完全释放内存。
• 我们知道哪个 UI 操作导致了高内存状况,但我们无法识别任何可疑代码。
问题
我们将不胜感激有关调试此类问题的任何建议。
WPF CommandManager 背景
_requerySuggestedHandlers
WPF CommandManager 拥有用于引发CanExecuteChanged
事件的 WeakReferences ( ) 的私有集合。处理CanExecuteChanged
成本相当高(尤其是为 找到 EventRoute CanExecute
,显然是 a RoutedEvent
)。每当 CommandManager 想要重新查询命令是否可以执行时,它都会遍历此集合并CanExecuteChanged
在相应的命令源上调用事件。
只要有被引用对象的 GC 句柄,WeakReferences 就不会从该集合中删除。虽然尚未收集对象,但 CommandHelper 会继续处理CanExecute
这些元素(ButtonBase 或 MenuItems)的事件。如果有很多垃圾(如我们的例子),这可能会导致大量调用 CanExecute 事件处理程序,从而导致应用程序非常滞后。