我正在尝试使用mef创建一个POC,我需要在一个准备好运行的项目中动态加载dll,为此我创建了一个控制台应用程序项目和一个类库
项目。
控制台应用程序项目的代码如下 -
namespace MefProjectExtension
{
class Program
{
DirectoryCatalog catalog = new DirectoryCatalog(@"D:\MefDll", "*.dll");
[Import("Method1", AllowDefault = true, AllowRecomposition = true)]
public Func<string> method1;
static void Main(string[] args)
{
AppDomainSetup asp = new AppDomainSetup();
asp.ShadowCopyFiles = "true";
AppDomain sp = AppDomain.CreateDomain("sp",null,asp);
string exeassembly = Assembly.GetEntryAssembly().ToString();
BaseClass p = (BaseClass)sp.CreateInstanceAndUnwrap(exeassembly, "MefProjectExtension.BaseClass");
p.run();
}
}
public class BaseClass : MarshalByRefObject
{
[Import("Method1",AllowDefault=true,AllowRecomposition=true)]
public Func<string> method1;
DirectoryCatalog catalog = new DirectoryCatalog(@"D:\MefDll", "*.dll");
public void run()
{
FileSystemWatcher sw = new FileSystemWatcher(@"D:\MefDll", "*.dll");
sw.NotifyFilter = NotifyFilters.CreationTime | NotifyFilters.Size;
sw.Changed += onchanged;
CompositionContainer container = new CompositionContainer(catalog);
container.ComposeParts(this);
Console.WriteLine(this.method1());
sw.EnableRaisingEvents = true;
Console.Read();
}
void onchanged(object sender, FileSystemEventArgs e)
{
catalog.Refresh();
Console.WriteLine(this.method1());
}
}
}
满足导入的库项目如下 -
namespace MefLibrary
{
public interface IMethods
{
string Method1();
}
public class CallMethods : IMethods
{
[Export("Method1")]
public string Method1()
{
return "Third6Hello";
}
}
}
一旦我编译了库项目(MefLibrary)并将dll放在D:\ MefDll位置并首次运行控制台应用程序,我将看到输出为
屏幕上的Third6hello
但现在如果我更改了method1的实现并使其返回“third7hello”构建MEF库项目并在D:\ MefDll处替换,而我的控制台应用程序即使在调用目录刷新打印后也运行onchanged处理程序 屏幕上的 Third6hello 而不是 third7hello
是否有人知道这是什么原因,如果是,请帮助。
答案 0 :(得分:8)
DirectoryCatalog.Refresh
只会添加新的或删除现有的程序集。它不会更新程序集。一个粗略的解决方法是:
DirectoryCatalog.Refresh
。这将删除程序集中包含的部分。DirectoryCatalog.Refresh
。这将添加程序集中包含的更新部件。注意:强>
AssemblyVersionAttribute
)。这是必需的,因为当使用DirectoryCatalog.Refresh
删除零件时,将不会卸载实际装配。只能在卸载整个应用程序域时卸载程序集。因此,如果DirectoryCatalog.Refresh
找到新程序集,它将使用程序集文件路径创建AssemblyCatalog
。然后AssemblyCatalog
会调用Assembly.Load
来加载程序集。但是,此方法不会加载与已加载的程序集具有相同AssemblyName.FullName
的程序集。CompositionOption.IsThreadSafe
标志创建CompositionContainer。正如我所提到的,这是一种解决方法。另一种方法是下载MEF's source code并使用DirectoryCatalog.cs作为您自己的目录编目实现的指南。
答案 1 :(得分:1)
在app域中加载dll后,无法从该域卸载。只能卸载整个域。因此,实现你所追求的并不容易。可以不断扫描更改并加载新副本并重新调用这些调用(您将以这种方式在域中累积越来越多无用的程序集),但我不相信这是MEF开箱即用的东西。换句话说,您正在观察的行为是设计的。
由于状态,执行此操作也很棘手且容易出错。想象一下,您在旧DLL的类实例中设置了一个字段并将其分配给变量。然后新的dll通过。旧实例及其字段会发生什么变化?显然,它们将保持不变,现在您在内存中使用了不同版本的插件正在使用。真糟糕!
如果您对此感兴趣,请注意why there isn't an Assembly.Unload method。并possible (conceptual) workaround。