使用void进行异步异常处理

时间:2011-06-08 19:41:25

标签: 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();
}

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

2 个答案:

答案 0 :(得分:17)

如果您从我的AsyncEx库中为您的控制台应用程序提供异步兼容的上下文(例如AsyncContextdocssource),那么您可以捕获传播出来的异常该上下文,甚至来自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();
}

答案 1 :(得分: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();
}

这将阻止每个DoSomething直到它完成,并在DoSomething投掷时退出循环。当然,那时你并没有真正做任何异步。但是从伪代码中,我不能完全告诉你想要异步发生什么。

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