有没有这样的写作方法:

public async Task<SomeResult> DoSomethingAsync()
{
    // Some synchronous code might or might not be here... //
    return await DoAnotherThingAsync();
}

而不是这样:

public Task<SomeResult> DoSomethingAsync()
{
    // Some synchronous code might or might not be here... //
    return DoAnotherThingAsync();
}

有意义吗?

为什么使用返回等待构造时,你可以直接返回任务<T>从内部DoAnotherThingAsync()调用?

我在很多地方看到等待返回的代码,我想我可能错过了一些东西。但据我所知,在这种情况下不使用async/await关键字,直接返回任务在功能上是等效的。为什么要增加额外等待层的额外开销?


当前回答

如果您不使用return await,那么在调试时或在异常日志中打印堆栈跟踪时,可能会破坏堆栈跟踪。

当您返回该任务时,该方法完成了它的目的,并退出了调用堆栈。 当你使用return await时,你将它留在调用堆栈中。

例如:

使用await时调用堆栈: A等待B的任务=>

不使用await时调用堆栈: A等待C的任务,B已经返回。

其他回答

如果你不需要async(即,你可以直接返回Task),那么不要使用async。

在某些情况下,return await是有用的,比如如果你有两个异步操作要做:

var intermediate = await FirstAsync();
return await SecondAwait(intermediate);

有关异步性能的更多信息,请参阅Stephen Toub关于该主题的MSDN文章和视频。

更新:我已经写了一篇博客文章来详细介绍。

你想要这样做的唯一原因是如果在前面的代码中有其他的等待,或者如果你在返回结果之前以某种方式操纵它。另一种可能发生这种情况的方式是通过try/catch改变异常的处理方式。如果你没有这样做,那么你是对的,没有理由增加使方法异步的开销。

非await方法的另一个问题是有时你不能隐式强制转换返回类型,特别是Task<IEnumerable<T>>:

async Task<List<string>> GetListAsync(string foo) => new();

// This method works
async Task<IEnumerable<string>> GetMyList() => await GetListAsync("myFoo");

// This won't work
Task<IEnumerable<string>> GetMyListNoAsync() => GetListAsync("myFoo");

错误:

不能隐式转换类型` System.Threading.Tasks.Task<System.Collections.Generic。列出>'到'System.Threading.Tasks.Task<System.Collections.Generic.IEnumerable>'

您可能希望返回await的另一个原因是:await语法允许您避免遇到Task<T>和ValueTask<T>类型之间的不匹配。例如,下面的代码即使SubTask方法返回Task<T>,但它的调用者返回ValueTask<T>。

async Task<T> SubTask()
{
...
}

async ValueTask<T> DoSomething()
{
  await UnimportantTask();
  return await SubTask();
}

如果你跳过DoSomething()行上的await,你会得到一个编译器错误CS0029:

不能隐式转换类型` System.Threading.Tasks。Task<BlaBla>'到'System.Threading.Tasks.ValueTask<BlaBla>'。

如果您尝试显式地类型转换它,也会得到CS0030。

顺便说一下,这是。net框架。我完全可以预见到一个评论说“这在。net的hypothetical_version中是固定的”,我还没有测试它。:)

将简单的“thunk”方法设置为异步会在内存中创建一个异步状态机,而非异步则不会。虽然这通常可以指向人们使用非异步版本,因为它更有效(这是真的),但这也意味着在挂起事件中,你没有证据表明该方法涉及“返回/延续堆栈”,这有时会使人们更难理解挂起。

所以,是的,当性能不是关键的时候(通常不是),我将在所有这些thunk方法上抛出async,这样我就有异步状态机来帮助我诊断挂起,并帮助确保如果这些thunk方法随着时间的推移而发展,它们一定会返回错误的任务,而不是抛出。