使用MEF加载带有嵌入式库的DLL

时间:2014-04-30 15:51:56

标签: c# .net dll resolver

我目前正在编写一个带有插件系统的应用程序套件,该系统使用MEF框架在运行时加载插件。

我目前设置了一个顶级WPF应用程序,将其作为嵌入式资源嵌入其中,并使用here描述的方法在运行时加载它们。

这很好用,我得到运行正常的单文件WPF应用程序。

然而,我的另一个顶级控制台应用程序使用MEF框架在运行时加载插件(WPF应用程序是固定的,并且显式包含插件)。我的插件本身在各种库上有几个依赖项,控制台应用程序加载插件的扩展文件夹中充斥着所有各种库dll。

我想在每个插件中嵌入每个插件的依赖项,以便我的扩展目录只包含顶级DLL文件。我上面使用的方法不适合这种方法,因为插件组件无法找到所需的依赖项,因为它只是正在搜索这些嵌入资源的执行程序集。

我当前的OnResolveAssembly方法如下所示:

public static Assembly OnResolveAssembly(object sender, ResolveEventArgs args)
{
    Assembly executingAssembly = Assembly.GetExecutingAssembly();
    var assemblyName = new AssemblyName(args.Name);

    string path = assemblyName.Name + ".dll";
    if (assemblyName.CultureInfo.Equals(CultureInfo.InvariantCulture) == false)
    {
        path = String.Format(@"{0}\{1}", assemblyName.CultureInfo, path);
    }

    using (Stream stream = executingAssembly.GetManifestResourceStream(path))
    {
        if (stream == null)
            return null;

        var assemblyRawBytes = new byte[stream.Length];
        stream.Read(assemblyRawBytes, 0, assemblyRawBytes.Length);

        return Assembly.Load(assemblyRawBytes);
    }
}

我认为最好的方法是添加功能以跟踪列表中加载的所有程序集,一旦以这种方式加载新程序集,递归地执行相同的操作;在你去的地方加载任何嵌入的DLL。然后,您可以将这些DLL添加到列表中,该列表将充当缓存。

还有更好的方法可以继续吗?

1 个答案:

答案 0 :(得分:2)

我已经为您实施了一个非常类似的解决方案,它对我来说非常好。如您所见,我在_references字典中跟踪已加载的程序集。

在我的情况下,我不需要急切地"以任何递归方式加载所有嵌入式依赖项,但我的嵌入式程序集确实按需向应用程序主机注册。

public static class ApplicationHost
{
    private static readonly Dictionary<string, Assembly> _references = new Dictionary<string, Assembly>();

    [STAThread]
    private static void Main()
    {
        AppDomain.CurrentDomain.AssemblyResolve += (sender, args) => _references.ContainsKey(args.Name) ? _references[args.Name] : null;
        RegisterAssemblyAndEmbeddedDependencies();
        // continue application bootstrapping...
    }

    public static void RegisterAssemblyAndEmbeddedDependencies()
    {
        var assembly = Assembly.GetCallingAssembly();
        _references[assembly.FullName] = assembly;
        foreach (var resourceName in assembly.GetManifestResourceNames())
        {
            using (var resourceStream = assembly.GetManifestResourceStream(resourceName))
            {
                var rawAssembly = new byte[resourceStream.Length];
                resourceStream.Read(rawAssembly, 0, rawAssembly.Length);
                var reference = Assembly.Load(rawAssembly);
                _references[reference.FullName] = reference;
            }
        }
    }
}