12

我正在使用 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();
}

并发生以下情况:“完成”被写入控制台,然后我在调试器中得到异常,然后我按 continue my program exists
是什么赋予了?

4

2 回答 2

29

如果您为您的控制台应用程序提供一个异步兼容的上下文(例如,我的 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();
}
于 2011-09-08T15:25:08.437 回答
15

当你调用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使用该异步方法。作为同步调用,它基本上只是安排工作,结果消失在以太中。

于 2011-06-08T20:00:14.853 回答