【发布时间】:2017-12-20 01:12:52
【问题描述】:
我的代码结合了异步方法和异常处理。代码很简单,所有任务都在等待,没有async void方法:
async Task DoWorkSafelyWithStateMachine()
{
try
{
await DoWorkThatMightThrowException();
}
catch (Exception exception)
{
Console.WriteLine("With state machine: " + exception.Message);
}
}
等待这个方法不会抛出异常,因为异常被吞噬了:
await DoWorkSafelyWithStateMachine(); // No exception thrown
但是,代码并不总是像预期的那样捕获异常。当方法以稍微不同的方式编写时出现问题,编译器没有创建异步状态机:
Task DoWorkSafelyWithoutStateMachine()
{
try
{
return DoWorkThatMightThrowException();
}
catch (Exception exception)
{
Console.WriteLine("Without state machine: " + exception.Message);
return Task.CompletedTask;
}
}
该方法没有用async 装饰,并且该方法内部没有等待任何内容。相反,try 中的方法返回的任务将返回给调用者。然而,根据我的经验,编译器的魔力仍然以某种方式确保如果try 中的方法抛出异常,它会被异常处理程序捕获。好吧,显然这并不总是正确的。
为了测试同一方法的这两种变体,我让DoWorkThatMightThrowException 抛出异常。如果方法不必在方法体中使用await,那么可以使用或不使用异步状态机来实现它:
async Task DoWorkThatMightThrowExceptionWithStateMachine()
{
throw new Exception("With state machine");
await Task.CompletedTask;
}
Task DoWorkThatMightThrowExceptionWithoutStateMachine()
{
throw new Exception("Without state machine");
return Task.CompletedTask;
}
我发现从DoWorkSafelyWithoutStateMachine 调用DoWorkThatMightThrowExceptionWithStateMachine 不会捕获抛出的异常。其他三个组合确实捕获了异常。特别是,在这两种方法中都没有涉及异步状态机的版本捕获了异常,我错误地推断了这个观察结果,不幸的结果是我的一些代码现在有细微的错误。
通过这个实验,我了解到我总是必须在try 块(表中的第一行)内执行await 任务。但是,我不明白表格第二行的不一致之处。请解释这种行为。它记录在哪里?搜索这方面的信息并不容易。由于未等待任务而导致异常“丢失”的更基本问题将主导搜索结果。
【问题讨论】:
-
你还没有问过问题。
-
@Servy:我认为 我不明白 是隐含的问题。但是,我稍微修改了措辞并添加了一个问号。
-
很高兴看到minimal reproducible example 使用上述方法以及它们是如何执行的。你的代码似乎......很奇怪。您可能会误解任务是如何管理其中引发的异常(可以这么说的)。如果不把所有部分放在一起很难说。
-
@Will:所有的代码都在那里。要执行它,您唯一要做的就是应用一些重命名来组合各种方法。有四个几乎相同的案例组合在一起,我不想将它们全部列出来复制粘贴,以免用代码夸大我的问题。
-
重点是标有'async'的方法(不管它是否有await语句)将被编译器重写为总是返回一个Task。如果抛出异常(甚至在第一次等待之前)-它将返回处于故障状态的任务,但仍然始终是一个任务。无论如何,在不等待的情况下将返回 Task 的内容包装在 try catch 中总是一个坏主意。
标签: c# .net exception-handling async-await try-catch