我正在使用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();
}
以下情况发生:“完成”被写入控制台,然后我在调试器中得到异常,然后按继续我的程序存在。
是什么给了什么?
答案 0 :(得分:17)
如果您从我的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();
}
答案 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
使用异步方法。作为同步调用,它基本上只是安排工作,结果消失在以太。