编辑 有关更多详细信息,请参阅问题底部的编辑说明。
原始问题
我有一个 CacheWrapper 类,它在MemoryCache
内部创建并保存 .NET 类的一个实例。
MemoryCache
将自身挂钩到 AppDomain 事件中,因此除非明确处置,否则它永远不会被垃圾收集。您可以使用以下代码验证这一点:
Func<bool, WeakReference> create = disposed => {
var cache = new MemoryCache("my cache");
if (disposed) { cache.Dispose(); }
return new WeakReference(cache);
};
// with false, we loop forever. With true, we exit
var weakCache = create(false);
while (weakCache.IsAlive)
{
"Still waiting...".Dump();
Thread.Sleep(1000);
GC.Collect();
GC.WaitForPendingFinalizers();
}
"Cleaned up!".Dump();
由于这种行为,我相信我的 MemoryCache 实例应该被视为非托管资源。换句话说,我应该确保它在 CacheWrapper 的终结器中被处理(CacheWrapper 本身就是 Disposable 遵循标准的 Dispose(bool) 模式)。
但是,我发现当我的代码作为 ASP.NET 应用程序的一部分运行时,这会导致问题。卸载应用程序域时,终结器在我的 CacheWrapper 类上运行。这反过来又会尝试处置该MemoryCache
实例。这是我遇到问题的地方。似乎Dispose
尝试从 IIS 加载一些配置信息,但失败了(可能是因为我正在卸载应用程序域,但我不确定。这是我的堆栈转储:
MANAGED_STACK:
SP IP Function
000000298835E6D0 0000000000000001 System_Web!System.Web.Hosting.UnsafeIISMethods.MgdGetSiteNameFromId(IntPtr, UInt32, IntPtr ByRef, Int32 ByRef)+0x2
000000298835E7B0 000007F7C56C7F2F System_Web!System.Web.Configuration.ProcessHostConfigUtils.GetSiteNameFromId(UInt32)+0x7f
000000298835E810 000007F7C56DCB68 System_Web!System.Web.Configuration.ProcessHostMapPath.MapPathCaching(System.String, System.Web.VirtualPath)+0x2a8
000000298835E8C0 000007F7C5B9FD52 System_Web!System.Web.Hosting.HostingEnvironment.MapPathActual(System.Web.VirtualPath, Boolean)+0x142
000000298835E940 000007F7C5B9FABB System_Web!System.Web.CachedPathData.GetPhysicalPath(System.Web.VirtualPath)+0x2b
000000298835E9A0 000007F7C5B99E9E System_Web!System.Web.CachedPathData.GetConfigPathData(System.String)+0x2ce
000000298835EB00 000007F7C5B99E19 System_Web!System.Web.CachedPathData.GetConfigPathData(System.String)+0x249
000000298835EC60 000007F7C5BB008D System_Web!System.Web.Configuration.HttpConfigurationSystem.GetApplicationSection(System.String)+0x1d
000000298835EC90 000007F7C5BAFDD6 System_Configuration!System.Configuration.ConfigurationManager.GetSection(System.String)+0x56
000000298835ECC0 000007F7C63A11AE System_Runtime_Caching!Unknown+0x3e
000000298835ED20 000007F7C63A1115 System_Runtime_Caching!Unknown+0x75
000000298835ED60 000007F7C639C3C5 System_Runtime_Caching!Unknown+0xe5
000000298835EDD0 000007F7C7628D86 System_Runtime_Caching!Unknown+0x86
// my code here
有什么已知的解决方案吗?我认为我确实需要MemoryCache
在终结器中处理它是正确的吗?
编辑
本文验证了 Dan Bryant 的回答,并讨论了许多有趣的细节。特别是,他介绍了 的情况StreamWriter
,它面临与我的情况类似的情况,因为它想在处理时刷新它的缓冲区。这篇文章是这样说的:
一般来说,终结器可能不会访问托管对象。然而,对于相当复杂的软件来说,对关闭逻辑的支持是必要的。Windows.Forms 命名空间使用 Application.Exit 处理此问题,它会启动有序关闭。在设计库组件时,有一种支持关闭逻辑的方法与现有的逻辑相似的 IDisposable 集成是很有帮助的(这避免了在没有任何内置语言支持的情况下定义 IShutdownable 接口)。这通常是通过在调用 IDisposable.Dispose 时支持有序关闭来完成的,并且在没有调用时支持中止关闭。如果可以使用终结器尽可能有序地关闭,那就更好了。
微软也遇到了这个问题。StreamWriter 类拥有一个 Stream 对象;StreamWriter.Close 将刷新其缓冲区,然后调用 Stream.Close。但是,如果 StreamWriter 未关闭,则其终结器无法刷新其缓冲区。微软通过不给 StreamWriter 终结器“解决”了这个问题,希望程序员会注意到丢失的数据并推断出他们的错误。这是需要关闭逻辑的完美示例。
综上所述,我认为应该可以使用 WeakReference 实现“托管终结”。基本上,让你的类在创建对象时注册一个对其自身的 WeakReference 和一个带有一些队列的 finalize 操作。然后,队列由后台线程或计时器监控,当它配对的 WeakReference 被收集时,它会调用适当的操作。当然,您必须小心,您的 finalize 操作不会无意中保留类本身,从而完全防止收集!