所以这是迄今为止的故事,我有这个工作者,它使用AppDomain来执行某些任务。该域名设置和拆卸都很昂贵。所以我为工作者创建了一个WeakReference对象的每个线程的缓存,如下所示:
class Worker
{
[ThreadStatic]
static Dictionary<string, WeakReference> _workers;
public static Worker Fetch( ... ) { you get the idea }
private AppDomain _domain;
public Worker(...)
{
_domain = AppDomain.Create( ... );
}
~Worker()
{
AppDomain.Unload(_domain);
}
// bla bla bla
}
我遇到的问题是,当GC收集时,似乎总是在调用AppDomain.Unload时引发异常:
System.CannotUnloadAppDomainException: Error while unloading appdomain. (Exception from HRESULT: 0x80131015)"
所以我觉得这很奇怪,我知道我在那个领域没有任何'跑步'......这笔交易是什么?我想出了一些挖掘和反复试验:
~Worker()
{
new Action<AppDomain>(AppDomain.Unload)
.BeginInvoke(_domain, null, null);
}
所以我的问题是:
答案 0 :(得分:11)
AppDomains由单独的CLR线程卸载。终结器线程运行时,该线程无法运行。你得到了异常,因为CLR注意到卸载线程没有进展。它永远不会发生,因为终结器线程在卸载调用时被阻止。
死锁。
您的解决方法确实解决了这个僵局。明确地卸载而不是依赖终结器是更好的方法。