有时我们的客户会在我们的应用程序中观察到内存不足异常。由于我们记录了他们的操作,我们可以大致重现他们所做的事情,但是如果我这样做并使用 dotMemory 分析应用程序,我无法重现异常并且使用的内存(大约 100 MB 托管 + 500MB 非托管)远低于限制(2GB,因为它是一个 32 位应用程序)。此外,在捕获异常时,使用 Process.GetCurrentProcess().WorkingSet64 请求当前内存使用量,这表明内存使用量在 500 到 900 MB 之间。我知道这个数字不是很可靠,但这是另一个迹象,表明应该有足够的可用内存。
该应用程序的一个相关属性是它处理测量的时间序列(DateTime 对和存储在数组中的双精度数)。这些对象可能足够大,可以存储在大对象堆 (LOH) 中。因此,确实会发生堆碎片,但在分析时这似乎没什么大不了的。LOH 的大小小于 100MB ,包括孔。
是否有可能在引发内存不足异常后调用垃圾收集器 (GC)?我认为,在内存分配请求不满足的情况下,只有当 GC 未能收集足够的内存时才会引发异常。但是,与在第 0 代堆中分配的内存相比,LOH 中分配的内存可能有所不同?
有谁知道,我们如何解决这个问题?
我们正在使用 VS 2010 SP1 和 .NET 4.0。该问题可能与此处、 此处和此处提出的问题有关,但我没有在那里找到令人满意的答案。
更新:添加了示例性堆栈跟踪和堆碎片图表
没有触发内存不足异常的唯一地方,但由于它被请求,我添加了一个 strack 跟踪:
Exception of type 'System.OutOfMemoryException' was thrown.
mscorlib
at System.Runtime.Serialization.ObjectIDGenerator.Rehash()
at System.Runtime.Serialization.ObjectIDGenerator.GetId(Object obj, Boolean& firstTime)
at System.Runtime.Serialization.Formatters.Binary.ObjectWriter.InternalGetId(Object obj, Boolean assignUniqueIdToValueType, Type type, Boolean& isNew)
at System.Runtime.Serialization.Formatters.Binary.ObjectWriter.Schedule(Object obj, Boolean assignUniqueIdToValueType, Type type, WriteObjectInfo objectInfo)
at System.Runtime.Serialization.Formatters.Binary.ObjectWriter.WriteMembers(NameInfo memberNameInfo, NameInfo memberTypeNameInfo, Object memberData, WriteObjectInfo objectInfo, NameInfo typeNameInfo, WriteObjectInfo memberObjectInfo)
at System.Runtime.Serialization.Formatters.Binary.ObjectWriter.WriteMemberSetup(WriteObjectInfo objectInfo, NameInfo memberNameInfo, NameInfo typeNameInfo, String memberName, Type memberType, Object memberData, WriteObjectInfo memberObjectInfo)
at System.Runtime.Serialization.Formatters.Binary.ObjectWriter.Write(WriteObjectInfo objectInfo, NameInfo memberNameInfo, NameInfo typeNameInfo, String[] memberNames, Type[] memberTypes, Object[] memberData, WriteObjectInfo[] memberObjectInfos)
at System.Runtime.Serialization.Formatters.Binary.ObjectWriter.Write(WriteObjectInfo objectInfo, NameInfo memberNameInfo, NameInfo typeNameInfo)
at System.Runtime.Serialization.Formatters.Binary.ObjectWriter.Serialize(Object graph, Header[] inHeaders, __BinaryWriter serWriter, Boolean fCheck)
at System.Runtime.Serialization.Formatters.Binary.BinaryFormatter.Serialize(Stream serializationStream, Object graph, Header[] headers, Boolean fCheck)
... <methods from our application follow>
dotMemory 的下图描述了使用该工具大约一个小时后的 LOH 碎片: