6

从主 AppDomain,我试图调用在不同 AppDomain 中实例化的类型中定义的异步方法。

例如,以下类型MyClass继承自MarshalByRefObject并在新的 AppDomain 中实例化:

public class MyClass : MarshalByRefObject
{
  public async Task<string> FooAsync()
  {
     await Task.Delay(1000);
     return "Foo";
  }
}

在主 AppDomain 中,我创建了一个新的 AppDomain 并在此 AppDomain 中创建了 MyClass 的一个实例,然后我调用了 async 方法。

var appDomain = AppDomain.CreateDomain("MyDomain");
var myClass = (MyClass)appDomain.CreateInstanceAndUnwrap(typeof(MyClass).Assembly.FullName, typeof(MyClass).FullName);
await myClass.FooAsync(); // BAM !

当然,SerializationException当我尝试进行调用时,我得到了一个,因为Task类型不继承自MarshalByRefObject,也不是可序列化的。

我怎么能解决这个问题?我真的很希望能够从另一个 AppDomain 中实例化的类型调用/等待异步方法......有没有办法?

谢谢 !

4

1 回答 1

0

我知道这个问题已有 7 年历史,但如果有人觉得这有帮助:

如今,一个简单的解决方案可能是调用同步方法Task.Run并等待该Task.Run方法。

而不是:

await RemoteObject.MethodAsync(); //Still throws a SerializationException.

利用:

await Task.Run(RemoteObject.Method); //If the method has no parameters.
await Task.Run(()=>RemoteObject.Method(parameter)); // If it does.

您可以获取方法的返回值(如果它不是 void,并且它的返回类型可以跨 AppDomains):

var result = await Task.Run(RemoteObject.GetDefaultValue); // If the method has no parameters.
var result = await Task.Run(()=>RemoteObject.GetValue(parameter)); // If it does.

Task.RunTask.Run<returnType>当委托具有非 void 返回类型时将隐式变为(委托本身将变为 aFunc而不是 a Action)。如果您使用方括号创建 lambda 表达式,那么您必须手动返回该方法的结果以暗示它是 aFunc而不是 a Action

var result = await Task.Run(()=>{ return RemoteObject.GetValue(parameter); });
于 2020-09-10T06:49:34.410 回答