MyAppDomain AssemblyResolve C#

时间:2015-01-28 13:58:16

标签: c# appdomain assembly-loading

我有一个DLL,我在一个环境中使用,我需要自己处理外部程序集的负载。我想将程序集加载到AppDomain中。当我使用CurrentAppDomain尝试此操作时,这工作正常,但我做到这个我自己创建的appdomain时失败了。 背景是我想在最后卸载appdomain,以便程序集被释放"最后。

public ZipEx() 
{
    try
    {
        AppDomainSetup domaininfo = new AppDomainSetup();
        domaininfo.ApplicationBase = System.Environment.CurrentDirectory;
        Evidence adevidence = AppDomain.CurrentDomain.Evidence;

        //THIS CODE WORKS
        //System.AppDomain.CurrentDomain.AssemblyResolve += CurrentDomain_AssemblyResolve;
        //System.AppDomain.CurrentDomain.Load("ICSharpCode.SharpZipLib");

        //THIS CODE DOES NOT WORK
        AppDomain zipDomain2 = AppDomain.CreateDomain("ADZib2", adevidence, domaininfo);
        PolicyLevel polLevel = PolicyLevel.CreateAppDomainLevel();
        PermissionSet permSet = new PermissionSet(PermissionState.Unrestricted);
        permSet.AddPermission(new SecurityPermission(SecurityPermissionFlag.AllFlags));
        polLevel.RootCodeGroup.PolicyStatement = new PolicyStatement(permSet);
        zipDomain2.SetAppDomainPolicy(polLevel);
        zipDomain2.AssemblyResolve += CurrentDomain_AssemblyResolve;
    }
    catch (Exception ex)
    {
        System.Windows.Forms.MessageBox.Show("ex in ctor" + Environment.NewLine + ex.ToString());
    }
    if (_loadedAssembly == null)
    {
    }
}

Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args)
{
    System.Windows.Forms.MessageBox.Show("CurrentDomain_AssemblyResolve");
    Assembly assembly = null;
    bool foundAssembly = false;

    int idx = args.Name.IndexOf(',');
    if (idx > 0)
    {
        string partialName = args.Name.Substring(0, idx);
        string dllName = partialName + ".dll";

        //Add the directorys where the assembly hould be resolved
        List<string> directorySearch = new List<string>
        {
          string.Format("{0}{1}{2}",Environment.CurrentDirectory,Path.DirectorySeparatorChar, dllName),
          string.Format("{0}{1}{2}",AppPath,Path.DirectorySeparatorChar, dllName)
        };

        foreach (string fileName in directorySearch)
        {
            if (File.Exists(fileName))
            {
                foundAssembly = true;
                assembly = Assembly.LoadFrom(fileName);
                break;
            }
        }

        if (assembly == null)
        {
            if (!foundAssembly)
            {
                foreach (string fileName in directorySearch)
                {
                }
            }
            else
            {
            }
        }
    }
    if (assembly != null) 
    {
        System.Windows.Forms.MessageBox.Show("assembly is not null");
    }
    return assembly;
}

我的问题是如何使用我创建的appdomain来加载程序集?

1 个答案:

答案 0 :(得分:1)

在您当前的示例中,我没有看到任何会导致新创建的AppDomain尝试加载程序集的未注释代码。我在此回答中假设您的未注释代码中缺少对“加载”的调用。

根据AppDomain.Load帮助中的备注,它只应在当前的AppDomain中使用,否则它会尝试将程序集加载到两个AppDomain中,这可能就是为什么你得到一个例外。要将程序集仅加载到其他AppDomain中,您应该调用其中一个CreateInstance函数。在这种特殊情况下,我建议使用CreateInstanceFromAndUnwrap函数,因为它允许您按名称和类型指定程序集。

如果您没有要实例化的类型并且能够跨AppDomain边界进行交互(这似乎不太可能但可能),您可能必须创建并丢弃一些简单类型,如枚举或结构让这个工作。