语法使用规则
await
必须且只能使用在async
函数中。async
函数体中如果没有await
关键字,则此函数永远不会被异步执行,而是和普通函数一样执行。- 异步代码使用任务的概念(
Task
和Task<T>
)对异步工作进行抽象。 - 作为一种命名传统,请为每个异步函数名添加 Async 后缀。
- 除非是winform程序中(事件处理),请避免使用
async void
,而应该使用async Task
或async Task<T>
。 - 如果使用
Task<T>
,await
将在任务完成后自动展开值。 - 被
await
修饰的只能是Task
或Task<T>
类型。
Asynchronous programming
Async in depth
彻底搞懂 C# 的 async/await
实际使用的经验教训
- 任务(tasks)只是对异步工作的抽象,而不是对线程的抽象。默认情况下,任务在当前线程上执行。如果要新开线程执行任务,使用
Task.Run();
。 - 但是使用
Task.Run();
开新线程做异步的开销很大,实测可以达到网络访问延迟的三分之二。 - 如果函数体中有
await Task.WhenAll()
, 则此函数应该由async Task
修饰, 而不是async void
。参考1 参考2 - 使用 Tuple 可以解决不能在
async
函数中使用out参数的问题:How to write an async method with out parameter?,如果参数超过7个,还可以使用嵌套 Tuple。 - 调用
.Result
属性获取返回值时,注意死锁问题。彻底搞懂 C# 的 async/await
C#异步的世界【下】
Task.WaitAll()
和 Task.WhenAll()
的区别
Task.WaitAll()
会阻塞当前线程直到所有任务完成;Task.WhenAll()
会创建一个新的任务,此任务只有在其他所有任务完成后才完成。- 由此可知,
Task.WaitAll()
是阻塞的方法,Task.WhenAll()
是非阻塞方法。
参考链接