C# async/await怎么使用 C#异步编程正确使用方法

来源:这里教程网 时间:2026-02-21 17:35:29 作者:

async/await 不是“让方法变快”的魔法开关,而是用来不阻塞线程、提升响应性和资源利用率的协作式异步模型。正确使用的关键在于理解“什么该异步”“谁在等待”“线程上下文怎么流转”。

只对真正异步的操作用 async/await

不是所有耗时操作都适合加 async。CPU 密集型任务(比如大数组排序、图像处理)用

Task.Run()
搬到线程池即可,不要盲目套 await;而 I/O 类操作(HTTP 请求、文件读写、数据库查询)天然支持异步,应优先使用它们的
Async
版本(如
HttpClient.GetAsync()
FileStream.ReadAsync()
)。

✅ 正确:调用
await httpClient.GetStringAsync(url)
❌ 错误:给普通 for 循环包一层
async Task
还 await 它
⚠️ 谨慎:CPU 工作用
await Task.Run(() => HeavyCalc())
,但要评估是否真有必要——可能直接同步执行更高效

async 方法必须有 await,且返回 Task 或 Task

标记为

async
的方法,编译器会重写为状态机。如果里面没写
await
,不仅失去异步意义,还会产生不必要的开销(装箱、状态机分配)。返回类型也必须匹配:

无返回值 →
async Task
(不是
void
,除非是事件处理器)
有返回值 →
async Task<string></string>
❌ 避免
async void
(除 UI 事件),它无法被 await、异常会直接崩掉线程

避免 .Result 和 .Wait(),防止死锁

在有同步上下文的环境(如 WinForms/WPF 主线程、ASP.NET 同步上下文旧版本),直接调用

task.Result
task.Wait()
极易引发死锁——因为 await 默认会尝试回到原上下文,而主线程正卡在等结果。

✅ 始终用
await task
✅ 如果真要同步等待(极少见),用
task.ConfigureAwait(false).GetAwaiter().GetResult()
放弃上下文捕获
❌ 不要混用:在 async 方法里又用 .Result

合理控制并发和异常处理

异步不是“放任不管”。多个异步操作并行时,注意资源竞争和异常聚合:

并发请求用
Task.WhenAll(tasks)
,别用循环 await(那是串行)
异常会被包装进
AggregateException
,建议用
try/catch
包住 await 表达式,直接捕获业务异常
需要取消时,传入
CancellationToken
并在 async 方法中适时检查或传给底层 API(如
HttpClient.GetAsync(url, token)

基本上就这些。async/await 本身不复杂,但容易忽略上下文、错误类型和调用链一致性。写完多问一句:这个 await 真的释放了线程吗?异常能被正确捕获吗?上层有没有在等它?

相关推荐