AppDomain.Unload会抛出Finalizer吗?

时间:2010-10-31 19:34:57

标签: c# appdomain finalizer

所以这是迄今为止的故事,我有这个工作者,它使用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);
    }

所以我的问题是:

  1. AppDomain.Unload会从终结器中失败吗?为什么?
  2. 我是否会在上述解决方法中遇到任何“不受欢迎的”?

1 个答案:

答案 0 :(得分:11)

AppDomains由单独的CLR线程卸载。终结器线程运行时,该线程无法运行。你得到了异常,因为CLR注意到卸载线程没有进展。它永远不会发生,因为终结器线程在卸载调用时被阻止。

死锁。

您的解决方法确实解决了这个僵局。明确地卸载而不是依赖终结器是更好的方法。