如何动态地将程序集加载到当前应用程序域到c#项目?

时间:2015-06-12 05:16:33

标签: c# .net .net-assembly

我尝试将第三方assemblies 动态加载到项目中,并使用reflection创建其类型的实例。

我用过:

Assembly.LoadFrom("Assembly1.dll")
Assembly.LoadFrom("Assembly2.dll")
Assembly.LoadFrom("Assembly3.dll") 

另外,尝试过:

AppDomain.CurrentDomain.Load("Assembly1.dll")
AppDomain.CurrentDomain.Load("Assembly2.dll")
AppDomain.CurrentDomain.Load("Assembly3.dll") 

但是,当我尝试创建其中一个类型的实例时,我不断收到The method is not implemented异常,如下所示:

Assembly.LoadFrom("Assembly1.dll")
Assembly.LoadFrom("Assembly2.dll")
Assembly assembly=  Assembly.LoadFrom("Assembly3.dll")
Type type=assembly.GetType("Assembly3.Class1")
object instance=Activator.CreateInstance(type); //throws exception at this point

但是,如果我在项目中直接add reference Assembly1,Assembly2和Assembly3 并执行:

Assembly3.Class1 testClass=new Assembly3.Class1();

我没有例外

我只是想知道我做错了什么?如何将程序集动态加载到项目中。我猜测,因为Class1实例的创建取决于另一个程序集Assembly1Assembly2,所以它失败了。那么,如何将所有依赖程序集动态加载到appdomain/loadcontext

非常感谢您的回答。

3 个答案:

答案 0 :(得分:1)

对于解析依赖关系,您需要处理AppDomain.AssemblyResolve Event

using System;
using System.Reflection;

class ExampleClass
{
    static void Main()
    {
        AppDomain ad = AppDomain.CurrentDomain;

        ad.AssemblyResolve += MyAssemblyResolveHandler;

        Assembly assembly = ad.Load("Assembly3.dll");

        Type type = assembly.GetType("Assembly3.Class1");

        try
        {
            object instance = Activator.CreateInstance(type);
        } 
        catch (Exception ex)
        {
            Console.WriteLine(ex.Message);
        }
    }

    static Assembly MyAssemblyResolveHandler(object source, ResolveEventArgs e) 
    {
        // Assembly.LoadFrom("Assembly1.dll")
        // Assembly.LoadFrom("Assembly2.dll")

        return Assembly.Load(e.Name);
    }
}

为每个程序集触发 MyAssemblyResolveHandler 而不是未加载,包括依赖项。

答案 1 :(得分:1)

使用" ad.AssemblyResolve + = MyAssemblyResolveHandler"时,我得到了' cdie'描述的无限循环。

所以我尝试了几件事。以下是通过MSDNs LoadFrom link

public object InitClassFromExternalAssembly(string dllPath, string className)
{
    try
    {
        Assembly assembly = Assembly.LoadFrom(dllPath);
        Type type = assembly.GetType(className);
        var instance = Activator.CreateInstance(type);
        return instance;
    }
    catch (Exception e)
    {
        Console.WriteLine(e);
        throw;
    }
}

显然,Assembly.LoadFrom方法需要DLL的完整路径。

请注意链接中通过LoadFrom加载装配体时可能出现的问题。

此外,' ArcangelZith'所包含的link上面有一些有趣的想法。

答案 2 :(得分:0)

如果引用了程序集,则可以按名称加载程序

AppDomain.CurrentDomain.Load(new AssemblyName("Assembly1.Core"))