触发并忘记异步任务方法有时不会调用

lan*_*gtu 6 .net c# asp.net-mvc fire-and-forget async-await

我有一个异步方法:

public async Task DoSomethingAsync(){
    ...
    await ...
    await ...
    ....
    return await SaveAsync();
}
Run Code Online (Sandbox Code Playgroud)

在大多数时候,我通过以下方式调用此方法:

await DoSomethingAsync()
Run Code Online (Sandbox Code Playgroud)

此调用按预期工作.但在某个地方,我需要将此方法称为火,并忘记:

public void OtherMethod()
{
    ...
    DoSomethingAsync(); //fire and forget here
}
Run Code Online (Sandbox Code Playgroud)

在这种情况下,有时Task会DoSomethingAsync()运行并完成,但有时候任务永远不会被调用(或调用某些await内部DoSomethingAsync()但从不完成最后一个await SaveAsync();).

我正在尝试通过以下代码确保以火灾和忘记方式调用任务:

public void OtherMethod()
{
    ...
    Task.Factory.StartNew(() =>
    {
        await DoSomethingAsync();
    }); //fire and forget again here
}
Run Code Online (Sandbox Code Playgroud)

但是,这不符合预期.所以我的问题是:

  1. 如何拨打DoSomethingAsync()无线电话await将始终运行并完成?(我不关心AppDomain重启/崩溃的情况)

  2. 如果我删除所有async/await代码DoSomethingAsync()并替换await.ContinueWith(),则调用Task DoSomethingAsync()(不具有async方法声明)将被调用并确定完成(忽略大小写AppDomain重启/崩溃),如果是,则从调用多长时间(我不如果在10分钟后调用任务,我会觉得我很开心)?

Ste*_*ary 5

你可能在某个地方遇到异常DoSomethingAsync,你无法观察到它,因为你忽略了这个任务.这正是您要求的行为,因为您告诉代码"忘记"任务.

要观察异常,你不能"忘记"任务:

public Task OtherMethodAsync()
{
  ...
  return DoSomethingAsync();
}
Run Code Online (Sandbox Code Playgroud)

并且在某些时候,await(或Wait)返回的任务.这就是你知道任务将运行和完成的方式.


Dan*_*ant 4

等待应该工作正常,因此这里可能发生其他事情阻碍了正在等待的一个或多个方法。有以下几种可能性:

  1. 您的方法之一中的某处出现死锁,导致其无法完成并阻止等待恢复。
  2. 由于某种原因,所有线程池线程都处于阻塞状态,从而阻止挂起的任务运行。
  3. 您正在同步上下文中运行异步方法,并且某些东西阻止了上下文,阻止它运行分派的回调。通常,上下文是 UI 线程,通常这是非常明显的,因为它会锁定您的应用程序 UI。

如果您可以连接 VS 调试器并观察发生了什么,请尝试暂停并查看并行堆栈视图。这应该有助于缩小考虑的可能性。


正如斯蒂芬指出的,当您称之为“即发即弃”时,也可能会发生异常。如果您还没有这样做,请确保处理TaskScheduler.UnobservedTaskException以记录此类事件。请注意,这是从终结器调用的,因此调用它的时间是不确定的。这可能会使调试变得棘手,因为该事件可能要晚于导致异常的实际事件才触发。因此,我建议遵循斯蒂芬的建议并保存任务以等待或稍后在其他地方等待。