5

我正在尝试追踪我的 .NET 应用程序中的内存泄漏。Windows 任务管理器报告内存使用量保持不变,而进程资源管理器报告内存使用量正在增加。

在任务管理器中,我正在查看唯一的内存列“内存(私有工作集)”。在 Process Explorer 中,我正在查看“Private bytes”列,因为它正在提高,而“Working set”下的值不是。

现在,当然 Process Explorer 是正确的,因为在多次分配后,我的应用程序因内存不足异常而崩溃。问题是,为什么任务管理器会误报应用程序的内存使用情况?不仅如此,它还误报了全局系统可用内存(性能选项卡中的图表保持不变)。

不需要我的代码,但这里是为了完整性。它显示了一个包含大数组的空窗口。当按下任意键时,窗口关闭并打开一个新窗口,其中包含一个新数组。旧窗口已经泄露,可能是由于 qt4dotnet GUI 库中的错误。

using System;
using com.trolltech.qt.gui;

namespace LeakTest
{
    class Test : QWidget
    {
        public byte[] Data = new byte[1000 * 1000 * 100];

        public Test()
        {
            show();
            GC.Collect(); // so measurements are more accurate
        }

        protected override void keyPressEvent(QKeyEvent arg__1)
        {
            disposeLater();
            new Test();
        }

        [STAThread]
        static void Main(string[] args)
        {
            QApplication.initialize(args);

            new Test();

            QApplication.exec();
        }
    }
}

操作系统:Windows 7

有趣的注意事项:当我将“数据”设为二维锯齿状维度数组时[1000 * 1000 * 100][1],任务管理器确实会报告提高内存使用率。

4

2 回答 2

8

它们是两种完全不同的记忆测量方法。工作集是您的程序使用的 RAM 量。它是一个不断变化的数字,受其他进程需要多少 RAM 的影响。您不能用完 RAM,Windows 通过将映射页面交换到页面文件来为您提供 RAM。

私有字节是您的程序使用的不与任何其他进程共享的虚拟内存量。在 32 位机器上,您有 2 GB 的可用虚拟内存。它需要在代码和数据之间共享。当可寻址的虚拟内存量中没有足够的空间来满足请求的分配时,您会得到 OOM。是的,这是更准确的数字。

一次请求 100 兆字节是有风险的。虚拟内存空间可能会变得碎片化,一段时间后可能仍然有大量可用的虚拟内存,但没有大到足以容纳 100 兆字节的空洞。锯齿状数组解决了这个问题,因为它是一个数组数组,所需的块要小得多,因此可以轻松容纳剩下的任何孔。

64位操作系统完全解决了这个问题。您的程序有许多千兆字节的可用地址空间,实际上仅受页面文件的最大大小限制。你根本无法用完大洞。

于 2011-02-04T20:14:46.037 回答
1

如果您想追踪内存泄漏,而您无法通过阅读代码找到它,那么您可能需要一个专为这项工作设计的真正工具。

任务管理器和进程资源管理器都不是调试内存泄漏的合适工具——你需要一些东西来告诉你在哪里分配了没有返回的内存。

于 2011-02-04T20:25:05.373 回答