在自己的AppDomain中加载和卸载程序集

时间:2013-10-26 14:46:29

标签: c# web-services

我做网络。通过输入参数加载程序集文件的服务。 然后在程序集中将尝试查找特定类型(继承自特定接口),创建实例并返回方法的结果。

我需要再次释放对程序集的调用。

从方法的输入参数中,我找到web.config中程序集的路径。并尝试加载它。

这是有效的代码:

[WebMethod]
    public String[] GetData(String confKey)
    {
        var assemblyPath = ConfigurationManager.AppSettings[confKey];
        var assembly = Assembly.LoadFrom(assemblyPath);

        List<String> retVals = new List<String>();

        foreach (var t in assembly.GetTypes())
        {
            if (t.ImplementsInterface(typeof(IMyServiceProvider)))
            {
                IMyServiceProvider objectInstance = Activator.CreateInstance(t) as IMyServiceProvider;
                retVals.Add(objectInstance.GetData());
            }
        }

        return retVals.ToArray();
    }

但是这样我可以删除加载的程序集或替换它,因为文件被“锁定”。

所以我尝试采用不同的方式将程序集加载到自己的AppDomain中,如下所示:

[WebMethod]
    public String[] GetData(String confKey)
    {
        var assemblyPath = ConfigurationManager.AppSettings[confKey];

        var tmp = String.Concat("AppDomain", Guid.NewGuid().ToString("N"));
        AppDomain dom = AppDomain.CreateDomain(tmp, null, AppDomain.CurrentDomain.SetupInformation);
        AssemblyName assemblyName = new AssemblyName();
        assemblyName.CodeBase = assemblyPath;
        Assembly assembly = dom.Load(assemblyPath);

        List<String> retVals = new List<String>();

        foreach (var t in assembly.GetTypes())
        {
            if (t.ImplementsInterface(typeof(IMyServiceProvider)))
            {
                IMyServiceProvider objectInstance = Activator.CreateInstance(t) as IMyServiceProvider;
                retVals.Add(objectInstance.GetData());
            }
        }

        AppDomain.Unload(dom);
        return retVals.ToArray();
    }

但是这个解决方案被抛出异常:

  

无法加载文件或程序集“NameOfMyAssembly”或其依赖项之一。给定的程序集名称或代码库无效。 (来自HRESULT的异常:0x80131047) - 在System.Reflection.AssemblyName.nInit(RuntimeAssembly&amp; assembly,Boolean forIntrospection,Boolean raiseResolveEvent)      在System.Reflection.RuntimeAssembly.CreateAssemblyName(String assemblyString,Boolean forIntrospection,RuntimeAssembly&amp; assemblyFromResolveEvent)      在System.Reflection.RuntimeAssembly.InternalLoad(String assemblyString,Evidence assemblySecurity,StackCrawlMark&amp; stackMark,IntPtr pPrivHostBinder,Boolean forIntrospection)      在System.Reflection.RuntimeAssembly.InternalLoad(String assemblyString,Evidence assemblySecurity,StackCrawlMark&amp; stackMark,Boolean forIntrospection)      在System.AppDomain.Load(String assemblyString)      在System.AppDomain.Load(String assemblyString)

为什么第一个解决方案程序集加载没有问题,第二个会抛出错误? 感谢

1 个答案:

答案 0 :(得分:1)

dom.Load(string)方法尝试按其全名加载程序集,该名称由程序集名称,版本,区域性信息和公钥组成。 CLR在GAC和应用程序的目录中搜索程序集。您还可以使用 .config 文件(app.config或)和探测元素设置其他搜索路径。根据我的理解, assemblyPath 是您要加载的程序集的路径,而不是作为参数传递的内容。我使用下一个代码在创建的域中加载程序集:

var domain = AppDomain.CreateDomain("Plugin domain"); // Domain for plugins
domain.Load(typeof(IPlugin).Assembly.FullName); // Load assembly containing plugin interface to domain 
// ...

var asm = Assembly.LoadFrom(pluginFile);
foreach (var exportedType in asm.GetExportedTypes())
{
    if (!typeof (IPlugin).IsAssignableFrom(exportedType))
        continue; // Check if exportedType implement IPlugin interface
    domain.Load(asm.FullName); // If so load this dll into domain
    // ...
}

由于所有插件位于“\ Plugins”目录下相对应用程序目录,我还添加了.config文件:

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
  <runtime>
    <assemblyBinding xmlns="urn:schemas-microsoft-com:asm.v1">
         <probing privatePath="Plugins"/>
    </assemblyBinding>
  </runtime>
</configuration>

请注意,如果您想使用仅指定CodeBase属性的 dom.Load(AssemblyName),则必须将其设置为Uri格式(即“file:/// D:/ Dir /。 ..“)我认为如果它们不在当前应用程序的文件夹中,则必须强烈命名这些程序集。

另外,我看到你是usig Activator.CreateInstance()来创建对象。在这种情况下,虽然您在创建的域中加载程序集,但您在当前域中创建对象,并且此对象的所有代码也在当前域中执行。要在单独的域中创建和执行代码,您应该使用 appDomain.CreateInstanceAndUnwrap()方法(link