使用void进行异步异常处理

TDa*_*ver 10 c# asynchronous exception-handling async-await c#-5.0

我正在使用Async CTP编写IO重型控制台应用程序.但我遇到异常问题.

public static void Main()
{
   while (true) {
     try{
         myobj.DoSomething(null);
     }
     catch(Exception){}
     Console.Write("done");
     //...
   }
}

//...
public async void DoSomething(string p)
{
   if (p==null) throw new InvalidOperationException();
   else await SomeAsyncMethod();
}
Run Code Online (Sandbox Code Playgroud)

并发生以下情况:"完成"被写入控制台,然后我在调试器中得到异常,然后我按继续我的程序存在.
是什么赋予了?

Ste*_*ary 18

如果您为您的控制台应用程序提供异步兼容的上下文(例如,来自我的AsyncEx库的AsyncContext(docs,source)),那么您可以捕获从该上下文传播的异常,即使是从async void方法:

public static void Main()
{
  try
  {
    AsyncContext.Run(() => myobj.DoSomething(null));
  }
  catch (Exception ex)
  {
    Console.Error.WriteLine(ex.Message);
  }
  Console.Write("done");
}

public async void DoSomething(string p)
{
  if (p==null) throw new InvalidOperationException();
  else await SomeAsyncMethod();
}
Run Code Online (Sandbox Code Playgroud)

  • 你已经恢复了对力量的平衡.谢谢! (5认同)

Arn*_*sen 11

当你打电话时,DoSomething()它基本上会创建一个Task引擎盖并启动它Task.由于你有一个void签名,没有任何Task对象可以发回信号,或者你可以阻止它,所以执行直接完成.与此同时,该任务抛出一个异常,没有人抓到,我怀疑,这是你的程序终止的原因.

我认为你想要的行为更像是这样的:

public static void Main()
{
   while (true) {
     var t = myobj.DoSomething(null);
     t.Wait();
     if(t.HasException) {
       break;
     }
   }
   Console.Write("done");
   //...
  }
}

//...
public async Task DoSomething(string p)
{
  if (p==null) throw new InvalidOperationException();
  else await SomeAsyncMethod();
}
Run Code Online (Sandbox Code Playgroud)

这将阻止每个DoSomething完成,如果DoSomething扔掉则退出循环.当然,那时你并没有真正做任何异步.但是从伪代码中,我不能完全告诉你想要异步发生什么.

主要内容:使用void异步方法意味着您无法获得异常,除非您await使用异步方法.作为同步调用,它基本上只是安排工作,结果消失在以太.