c# - AppDomain 卸载杀死父 AppDomain

标签 c# .net-3.5 appdomain

我无法弄清楚有关我的 AppDomain.Unload(...) 调用的问题。我有来自 my earlier question 的代码的详细解释.事实证明,我执行了几个显然不需要的步骤。但是,我相当确定当创建 AppDomain 并将其保存在集合中时:

private static Dictionary<string , AppDomain> HostDomains;

void StartNewDomain(string domainName)
{
    AppDomain domain = AppDomain.CreateDomain(domainName);
    HostDomains[domainName] = domain;
}

...完成后,您必须卸载它:

if (HostDomains.ContainsKey(domainName))
{
    AppDomain.Unload(HostDomains[domainName]);
    HostDomains.Remove(domainName);
}

然后从集合中删除域。

然而,当我卸载域时,整个应用程序将结束。如果我删除卸载,一切都很好......我们只剩下从集合中删除域。但我担心我的子 AppDomain 没有真正卸载。我猜它最终可能会得到 GC,但这并没有给我一个温暖的模糊。

子 AppDomain 程序集(一个 Windows 窗体应用程序)通过我的继承 MarshalByRefObject 的适配器类中引用的接口(interface) (IModule) 异步启动。我想知道对 IModule 的 Start() (插件模块程序集实现)的引用是否没有正确编码(因为我的实现)。所以,当 Shutdown() 方法被调用时,整个应用程序就死掉了。我是否应该让我的 IModule 成为一个抽象类,以便它也应该继承 MBR?不解...

查看我的代码后:

// instances the module for access to the module's Start() method
    IModule module = (IModule)domain.CreateInstanceAndUnwrap(
    ModuleManager.Modules[modName].Name, 
    ModuleManager.Modules[modName].EntryPoint.FullName);

...我担心的是,由于 IModule 是一个接口(interface),即使我在子域中创建一个实例,程序集也会泄漏到我的主 AppDomain 中。因此,当我尝试卸载子域时,两个域都被卸载了。这是正确的吗?通过 MBR(适配器)对象提供 Start() 和 Stop() 方法的最佳解决方案可能是什么?

更新:请参阅下面我的回答以了解更改 --
好的,没有泄漏——一切都继承了 MBR:

  1. Host : MarshalByRefObject -- 在新的 AppDomain 中实例化 ModuleAdapter
  2. ModuleAdapter : MarshalByRefObject -- IModule 接口(interface),接口(interface)方法 (Start,Stop)
  3. MyModulePlugin : MarshalByRefObject -- Application.Run(myForm)

我还是做错了什么吗?我已经尝试了几件事,但它似乎是错误的或不完整的。当我告诉 ModuleAdapter 关闭时,它会调用 AppDomain.Unload(AppDomain.CurrentDomain) 并且主机域也会停止。我仍然在应用程序退出时遇到一些第一次机会异常。但是表单 (myForm) 已被告知 .Close()。

所以,我仍在寻找正确的方法...

最佳答案

正如我所怀疑的,在主域中使用 IModule 接口(interface)实例化会导致泄漏。为了正确地做到这一点:

AppDomain domain = AppDomain.CreateDomain(domainName);
HostDomains[domainName] = domain;  // put in collection

ModuleAdapter adapter = (ModuleAdapter)domain.CreateInstanceAndUnwrap(asmName , typeName);

其中 ModuleAdapter 继承 MarshalByRefObject。然后:

adapter.Execute(moduleAssembly , moduleType);

在 ModuleAdapter 类中:

public void Execute(string Name, string EntryPoint)
{
    module  = (IModule)AppDomain.CurrentDomain.CreateInstanceAndUnwrap(Name , EntryPoint);
}

我确实欢迎评论或其他答案以获得更好的方法。

将实例移至 ModuleAdapter 类后,我们仍然遇到 AppDomain.Unload 杀死整个应用程序的问题。我想知道这是否是因为在模块插件中我们正在使用 Application.Run(myForm) - 然后当我们关闭时我们调用 myForm.Close()。显然这会关闭应用程序,所以我想知道 myForm.Close() 是否也“卸载”了 AppDomain。

关于c# - AppDomain 卸载杀死父 AppDomain,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2294474/

相关文章:

c# - 从 asp.net 中的文本框获取值

java - 在这种情况下,appdomains 是否有帮助?

c# - .net 4.0 安全模型。另一个 "Inheritance security rules violated by type..."异常

.net-3.5 - 扩展方法命名空间和赞助商类的命名约定

c# - 我可以从另一个 AppDomain 实例化一个类型为 'dynamic' 吗?

c# - WebBrowser Control 导致整个应用程序无响应

c# - Visual Studio 2019 : Cannot get rid of warning on VC++ 2015 runtime not found

C# 正则表达式匹配字母、数字、下划线、破折号和点

c# - 隐藏继承成员

c# - 在接口(interface)中定义可选的实现方法?