从主 AppDomain 订阅 DomainUnload 事件

Subscribe to DomainUnload event from a main AppDomain

我正在创建一个与可创建和卸载的新 AppDomain 一起工作的应用程序。我想处理 DomainUnload 事件,所以我需要订阅它。我无法从可以卸载的新 AppDomain 订阅此事件,因为卸载域后将无法访问那里的所有成员。

但是,当我尝试从 Main AppDomain 订阅此事件时,我总是遇到异常:“System.Runtime.Serialization.SerializationException:程序集 'MyMainAssembly' 中的类型 'MyNamespace.MainWindow' 不是标记为可序列化。"

最初我尝试在主应用程序域中向 class 添加属性 [Serializable],但没有帮助。然后我创建了一个从 MarshalByRefObject 派生的新 class 'RemoteHandlerClass' 并将订阅代码放入其中,但我也遇到了同样的错误。在此 class 中使用 [Serializable] 属性也没有帮助。要从此 class 访问成员,我在主应用程序域中使用代理对象,创建者:

var remoteHandler = (RemoteHandlerClass)workerDomain.CreateInstanceAndUnwrap(
    typeof(RemoteHandlerClass).Assembly.FullName, typeof(RemoteHandlerClass).FullName);

如何订阅此活动?最好我想从主应用程序域订阅此事件。下面的代码可以重现我的问题。

谢谢!

public partial class MainWindow : Window
{
    private void Button_Click(object sender, RoutedEventArgs e)
    {
        var newDomain = AppDomain.CreateDomain("New domain");
        newDomain.DomainUnload += UnloadHandler; // System.Runtime.Serialization.SerializationException:
                                                 // Type 'MyNamespace.MainWindow' in assembly 'MyMainAssembly'
                                                 // is not marked as serializable.
    }
    
    private void UnloadHandler(object sender, EventArgs e)
    {
        
    }
}

因此,在处理 AppDomains 时,您的处理程序对象需要是 MarshalByRefObject 而不是 - 它只是一个 Window 类型。用简单的 class 包装您正在做的事情,例如

class DomainContainer : MarshalByRefObject
    {
        public void StartDomain()
        {
            var newDomain = AppDomain.CreateDomain("New domain");
            newDomain.DomainUnload += UnloadHandler;
            
        }

        private void UnloadHandler(object sender, EventArgs e)
        {

        }
    }

并在您的应用中调用它。

 private void Button_Click(object sender, RoutedEventArgs e)
        {
            var d = new DomainContainer();
            d.StartDomain();
        }