.net - 为什么从不调用 NET 垃圾收集器?

标签 .net garbage-collection finalizer

我正在开发一个内存消耗有问题的应用程序。如果用户在应用程序中单击足够长的时间,它会以 OutOfMemoryException 结束。

我用“ANTS Memory Profiler”对应用程序进行了很长时间的分析,在我看来没有“经典”内存泄漏(例如,阻止对象被垃圾收集的事件处理程序)。

但是所有保留在内存中的对象都有一个共同点——它们直接或间接地使用一个实现终结器的标准 .NET 控件(例如 TextBox、Numberbox)。在“ANTS Memory Profiler Instance Retention Graph”中,我可以看到唯一持有控件引用的实例是 .NET Finalizer Queue。

链接保留图(我没有足够的声誉直接发布图像:-))--> http://i50.tinypic.com/2d6r6nn.png

因此我调查了终结器线程中死锁的方向
(参见 http://dotnetdebug.net/2005/06/22/blocked-finalizer-thread/)
但找不到死锁的迹象。此外,与死锁理论相反的是,在使用内存分析器进行内存快照后,触发 GC.Collect(), View 被垃圾收集 - 分别执行它们的终结器并且一切正常。

所以,这看起来像是带有 Finalizer 的 .net 对象的正常生命周期,对吧?但是在我的应用程序中,我可以单击直到 OutOfMemoryException 并且垃圾收集器永远不会运行!

处理这个问题的最后一次尝试是使用 GC.AddMemoryPressure(),因为在 View 中有很多位图,它们分配了很多非托管代码。但这也不能激发垃圾收集器收集空闲内存。

所以,我认为应用程序中的一个概念存在本质上的错误,它阻止了 GC 释放内存,但我完全不知道是什么。

有没有人经历过类似的经历并有任何线索?

此致

和我

最佳答案

OutOfMemoryException有时说谎。有时它的意思是“我无法获得非托管句柄”——它实际上并不总是与内存相关。问题是通常很难判断为什么会失败,“内存不足”可能是一个合理的猜测。

好像很多事情没有及时处理。处理它们会调用它们的自定义代码来急切地主动释放它们未损坏的句柄。

相反,GC 主要是由内存压力触发的。如今,PC 可能有大量内存可用,但会耗尽非托管句柄。

关于.net - 为什么从不调用 NET 垃圾收集器?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15342875/

相关文章:

Java vs C# 多线程性能,为什么 Java 变慢了? (包括图表和完整代码)

java - "Cannot draw recycled bitmaps"如果我在 onDestroy() 中调用 recycle()

c# - 终结器陷入无限循环

c# - 从 List<T> 创建 MVC3 CheckBoxFor 并在 Post 上取回列表(具有更新的值)

.net - 如何判断数组类型?

performance - 内存分配/释放瓶颈?

c# 析构函数(和终结器?)在程序结束时自动调用 - 我应该对此做些什么吗?

node.js - 当我的对象即将被 Node 中的 GC 收集时,我可以得到回调吗?

c# - 如何在程序集中找到所有类,这些类是通用抽象类的实例并实现某个接口(interface)

java - 线程池工作线程被 Runnables 淹没导致 JVM 崩溃