我有一个应用程序,其中包含许多连接到许多不同I / O设备的插件(MEF)。这些插件中的大多数都有许多托管和非托管dll。
一家制造商最近发布了新固件和新驱动程序。 API保持不变。
我认为我可以在单独的资源文件夹中包含这两个dll版本,并在应用程序启动时将所需的集复制到输出文件夹中。
(我想我可以制作插件的第二个副本并找出加载正确插件的方法,但我认为复制DLL可能会更容易 - 特别是考虑到插件代码没有改变)
这不起作用。
static MyClass() // static constructor
{
// required version?
if (envvar.Equals("4") )
{
path = "ver4.1.1";
}
else
{
path = "ver2.5.1";
}
// location of drivers
path = Path.Combine("./Prj/QX", path);
string[] files = Directory.GetFiles(path);
// Copy the files and overwrite destination files if they already exist.
foreach (string s in files)
{
string fileName = Path.GetFileName(s);
string destFile = Path.Combine(".", fileName);
File.Copy(s, destFile, true);
}
// force load of assemby
Assembly assy = LoadWithoutCache("driver.dll");
string fn = assy.FullName;
}
static Assembly LoadWithoutCache(string path)
{
using (var fs = new FileStream(path, FileMode.Open))
{
var rawAssembly = new byte[fs.Length];
fs.Read(rawAssembly, 0, rawAssembly.Length);
return Assembly.Load(rawAssembly);
}
}
错误消息表明原始DLL已加载或应该已加载但不能加载。
Could not load file or assembly 'driver, Version=1.5.77, Culture=neutral, PublicKeyToken=983247934' or one of its dependencies. The located assembly's manifest definition does not match the assembly reference. (Exception from HRESULT: 0x80131040)
有没有办法实现我的目标,还是我必须构建我的应用程序的2个单独版本?
修改 我还应该说,没有必要同时加载两个DLL。用一个参数启动应用程序就足够了,这个参数会导致加载一个或另一个DLL。
答案 0 :(得分:4)
实现这一目标的一种(复杂)方法是:
FSW检测程序集是否被更改或删除。
只能检查加载到此上下文中的程序集! 通过获取程序集的类型并检查类型是否实现PluginInterface,确保程序集类型实现PluginInterface。
由于您只能使用其所有程序集卸载AppDomain,因此您需要在其自己的AppDomain中加载每个插件程序集。这样就可以卸载插件的AppDomain。
请务必终止插件的所有进程/线程!
卷影复制通过将程序集复制到临时路径来确保更改原始文件。
将成功加载的程序集的路径放入列表中。 这样可以更容易地检测装配是否需要加载/重新加载。
检查更改/删除哪个.dll以卸载程序集,或检测到新程序集加载它。
问候, Blackanges
答案 1 :(得分:1)
You could do it in several ways.一种可能性是使用Managed Extensibility Framework (MEF)。 您甚至可以监视dll所在的文件夹,并在新版本可用时立即加载新版本。
另一种可能性是您可以使用reflection在运行时动态加载所需的dll。
答案 2 :(得分:0)
也许你可以通过这样的反射加载你的dll:Loading DLLs at runtime in C#
但我怀疑它是否适用于更大的项目