0

我有一个函数将生成一个数据集并从中创建一个 xml 文件。该功能运行良好。问题是多次运行报告后会出现“内存不足”错误。当我测试报告时,我发现当它到达这个 XmlDocument 命令时,内存使用量会增加很多。我用了GC但没用,有什么建议吗?

if (ddlDir.SelectedItem != null && ddlSec.SelectedItem != null)
{
    using (DataSet dsClosedKPICalls = GetReportData())
    {
        dsClosedKPICalls.DataSetName = "ClosedKPICalls";
        foreach (DataTable table in dsClosedKPICalls.Tables)
        {
            table.TableName = "ServiceInfo";
        }
        XmlContent = dsClosedKPICalls.GetXml();
    }

    XmlDocument XML_Data = new XmlDocument();           // contains the resultant XML data

    XML_Data.LoadXml(XmlContent);
    XmlNodeList TablesList = XML_Data.SelectNodes("ClosedKPICalls/ServiceInfo");

    for (int i = 0; i < TablesList.Count; i++)
    {
        XmlDocument innerXML = new XmlDocument();

        using (DataSet dsTaskDetails = getSplitupRecords(TablesList.Item(i).SelectSingleNode("ServiceNo").InnerText))
        {
            dsTaskDetails.DataSetName = "TaskDetails";
            foreach (DataTable tbl in dsTaskDetails.Tables)
            {
                tbl.TableName = "RequestInfo";
            }

            innerXML.LoadXml(dsTaskDetails.GetXml());

            TablesList.Item(i).AppendChild(XML_Data.ImportNode(innerXML.SelectSingleNode("TaskDetails"), true));

            innerXML = null;
            GC.Collect();
            GC.WaitForPendingFinalizers();
        }
    }
}
4

1 回答 1

2

您的 XML 文档有多大?具体来说,它们是否大于约 85K?

.Net 框架使用称为大对象堆 (LOH)的东西,它将用于大于 ~85K 的分配。如果您的 XML 文档比这大,那么东西(如字符串)可能会在 LOH 中分配。LOH 的问题在于它没有被压缩,因此它容易受到内存碎片的影响,这反过来可能导致OutOfMemoryExceptions即使进程通常应该能够分配该内存也被抛出。这可能是您的问题的原因。

对于解决方案,您可以尝试

  • 通过尝试将对象的大小保持在 85K 以下来减少 LOH 的使用。根据您的应用程序,这可能很难实现,但是可以(例如)将大型 XML 文档拆分为许多较小的文档。
  • 尝试以 .Net framework v4.5 为目标,看看是否有帮助。.Net 框架 v4.5 包括对 LOH 的改进,这可能在这种情况下有所帮助。

GC.Collect顺便说一句,您对and的调用GC.WaitForPendingFinalizers是一个坏主意,因为这可以人为地将低代对象提升为高代对象,并且通常会“与垃圾收集器混淆”。在这种情况下,这些调用几乎肯定也无济于事。作为一般规则,您应该避免手动调用垃圾收集,除非您确切知道自己在做什么。

于 2012-09-25T13:46:09.433 回答