23

我需要在运行时加载的程序集中执行一个方法。现在我想在方法调用之后卸载那些加载的程序集。我知道我需要一个新的 AppDomain 以便我可以卸载这些库。但是在这里,问题出现了。

要加载的程序集是我的插件框架中的插件。他们根本没有切入点。我所知道的是它们包含一些实现给定接口的类型。旧的非 AppDomain 代码如下所示(略微缩短):

try
{
    string path = Path.GetFullPath("C:\library.dll");
    AppDomain.CurrentDomain.AssemblyResolve += CurrentDomain_AssemblyResolve;
    Assembly asm = Assembly.LoadFrom(path);
    Type[] types = asm.GetExportedTypes();
    foreach (Type t in types)
    {
        if ((t.GetInterface("IStarter") != null) && !t.IsAbstract)
        {
            object tempObj = Activator.CreateInstance(t);
            MethodInfo info = t.GetMethod("GetParameters");
            if (info != null)
            {
                return info.Invoke(tempObj, null) as string;
            }
        }
    }
}
catch (Exception ex)
{
    MessageBox.Show(String.Format("Damn '{0}'.", ex.Message), "Exception", MessageBoxButtons.OK, MessageBoxIcon.Error);
}

private Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args)
{
    if (args.Name.StartsWith("MyProject.View,"))
    {
        string path = Path.GetFullPath("C:\view.dll"));
        return Assembly.LoadFrom(path);
    }
    return null;
}

现在我希望它们加载到自己的 AppDomain 中:

try
{
    string path = Path.GetFullPath("C:\library.dll");
    AppDomain domain = AppDomain.CreateDomain("TempDomain");
    domain.AssemblyResolve += CurrentDomain_AssemblyResolve;  // 1. Exception here!!
    domain.ExecuteAssembly(path);  // 2. Exception here!!
    domain.CreateInstanceFrom(...);  // 3. I have NO clue, how the type is named.
    domain.Load(...);  // 4. I have NO clue, how the assembly is named.
    domain.DoCallBack(...); // 5. Exception here!!
    // ...
}
catch (Exception ex)
{
    MessageBox.Show(String.Format("Damn '{0}'.", ex.Message), "Exception", MessageBoxButtons.OK, MessageBoxIcon.Error);
}

如您所见,我已经放入了 5 个案例。

  1. 如果我设置事件处理程序,我会得到一个异常,即程序集(它是一个管理控制台 (mmc.exe) SnapIn。无法找到/加载。

  2. ExecuteAssembly 没有找到入口点(嗯,没有)。

  3. 我不知道该类型是如何命名的。如何通过界面加载?

  4. 类似于 3. 如何获取程序集的名称?

  5. 与 1 中的错误相同。

我认为问题可能出在管理控制台上,或者我不知道我做错了什么。任何帮助表示赞赏。

更新 1

我现在尝试使用发布的代理解决方案。

AppDomain domain = AppDomain.CreateDomain("TempDomain");
InstanceProxy proxy = domain.CreateInstanceAndUnwrap(Assembly.GetAssembly(
    typeof(InstanceProxy)).FullName, typeof(InstanceProxy).ToString()) as InstanceProxy;
if (proxy != null)
{
    proxy.LoadAssembly(path);
}
AppDomain.Unload(domain);

public class InstanceProxy : MarshalByRefObject
{
    public void LoadAssembly(string path)
    {
        AppDomain.CurrentDomain.AssemblyResolve += CurrentDomain_AssemblyResolve;
        Assembly asm = Assembly.LoadFrom(path);
        Type[] types = asm.GetExportedTypes();
        // ...see above...
    }
}

这也不起作用。尝试创建代理对象时,出现异常:

无法加载文件或程序集“MyProject.SnapIn,Version=1.0.0.0,Culture=neutral,PublicKeyToken=null”或其依赖项之一。该系统找不到指定的文件。

错误消息中的文件是加载到 mmc(管理单元)中的文件。知道如何解决此错误吗?AppDomain.AssemblyResolve 未被调用(在旧域或新域中均未调用)。

更新 2

我现在已经尝试了使用 AppDomainSetup 的解决方案。现在,异常已更改为:

无法加载文件或程序集“file:///C:/Development/MyProject/bin/SnapIn/MyProject.SnapIn.DLL”或其依赖项之一。给定的程序集名称或代码库无效。(来自 HRESULT 的异常:0x80131047)

任何的想法?

4

3 回答 3

15

试试这个:

namespace SeperateAppDomainTest
{
    class Program
    {
        static void Main(string[] args)
        {
            LoadAssembly();
        }

        public static void LoadAssembly()
        {
            string pathToDll = Assembly.GetExecutingAssembly().CodeBase;
            AppDomainSetup domainSetup = new AppDomainSetup { PrivateBinPath = pathToDll };
            var newDomain = AppDomain.CreateDomain("FooBar", null, domainSetup);
            ProxyClass c = (ProxyClass)(newDomain.CreateInstanceFromAndUnwrap(pathToDll, typeof(ProxyClass).FullName));
            Console.WriteLine(c == null);

            Console.ReadKey(true);
        }
    }

    public class ProxyClass : MarshalByRefObject { }
于 2010-01-25T14:35:15.670 回答
1

看看这个先前的答案:How to load an assembly into different AppDomain on Windows Mobile (.NET CF) ?. 该答案会创建一个代理类,该类会运行到新的 AppDomain 上下文中,因此您可以在那里完全控制初始化。

您可以在类中创建一个Start()方法ServiceApplicationProxy,然后使用proxy.Start().

于 2010-01-25T13:59:22.827 回答
0

https://msdn.microsoft.com/en-us/library/3c4f1xde%28v=vs.110%29.aspx

指定

typeName 类型:System.String

The fully qualified name of the requested type, including the namespace but not the assembly, as returned by the Type.FullName

财产。

所以尝试使用完全限定名称调用,而不是 typeof(InstanceProxy).ToString()使用字符串/文本"<<Namespace>>.InstanceProxy"

如下

InstanceProxy proxy = domain.CreateInstanceAndUnwrap(path, "<<Namespace>>.InstanceProxy") as InstanceProxy;
于 2016-11-24T14:28:46.473 回答