我做网络。通过输入参数加载程序集文件的服务。 然后在程序集中将尝试查找特定类型(继承自特定接口),创建实例并返回方法的结果。
我需要再次释放对程序集的调用。
从方法的输入参数中,我找到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)
为什么第一个解决方案程序集加载没有问题,第二个会抛出错误? 感谢
答案 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)