我不完全确定该去哪里寻求帮助,所以我想我会尝试使用 stackoverflow,因为它通常可以回答我大约 90% 的编程相关问题。
简而言之,我有一个正在泄漏内存的开源 .NET 应用程序。从某种意义上说,这可能不是真正的内存泄漏,当应用程序关闭时,我怀疑内存已被回收,但在它运行时,它会不断分配更多内存而不释放它。最终,aSystem.OutOfMemoryException
被抛出。
为了调试问题,我按照本文中推荐的步骤进行了调试,并生成了下图,其中红色是 .NET/CLR 内存“#Bytes in all Heaps”,绿色是带有 Windows 性能监视器工具的进程“Private Bytes” (请注意,绿线已被统一缩小,看起来更接近红线,因为只有线条的形状对我来说很重要):性能监视器输出。
我将图像作为托管内存泄漏的证据,然后使用 Windows 的调试诊断工具尝试定位泄漏源(如文章中所述)。然而,我从调试诊断工具得到的报告非常奇特。
基本上,每次尝试在应用程序运行时每 5 秒收集一次“完全 UserDump”,但由于当时垃圾收集器始终处于垃圾收集周期的中间,导致调试诊断输出错误并阻止它收集任何有用的 .NET 内存相关信息的工具。
现在我被卡住了,我知道我有一个托管的内存泄漏,但我不知道如何缩小它的范围。我也很困惑垃圾收集器如何总是处于收集周期的中间,这让我想知道垃圾收集线程是否被某种方式阻塞,阻止它释放内存和/或退出垃圾收集周期。
在性能监视器图的某些部分中,分配的 .NET 内存会下降一点,因此垃圾收集器不会永远卡住,但它肯定是大部分时间都卡住了,否则调试诊断应该能够做到一个用户转储。
几个问题:
.NET 应用程序中的垃圾收集器是否有可能在尝试释放一些内存时卡住,可能来自一些编码不良的析构函数/终结器或其他东西?
我可以使用什么策略来继续缩小问题的根源?