async和await用于简化异步编程,避免阻塞主线程。方法用async修饰,返回Task或Task<T>,通过await等待异步操作,支持异常处理和并发执行,需避免使用.Result以防死锁,广泛应用于I/O操作。

在C#中,async 和 await 是异步编程的核心关键字,它们让编写异步代码变得像写同步代码一样简单。使用 async/await 可以避免阻塞主线程(比如ui线程),提高程序响应性和性能,尤其是在处理I/O操作(如网络请求、文件读写、数据库查询)时非常有用。
1. 基本语法和使用规则
要使用 async 和 await,需遵循以下基本结构:
public async Task<int> DownloadDataAsync()
{
HttpClient client = new HttpClient();
String result = await client.GetStringAsync(“https://example.com”);
return result.Length;
}
说明:
- 方法用 async 修饰,表示这是一个异步方法。
- 方法返回类型通常是 Task 或 Task<TResult>,也可以是 void(不推荐用于普通方法)。
- 在需要等待异步操作完成的地方使用 await,它会挂起当前方法的执行,而不阻塞线程,等任务完成后继续执行后续代码。
2. 常见返回类型说明
异步方法支持以下几种返回类型:
- Task:用于没有返回值的异步操作,类似 void 方法。
- Task<T>:用于有返回值的异步操作,T 是返回的数据类型。
- void:仅用于事件处理程序。不建议普通方法使用,因为无法被 await,异常也难以捕获。
示例:
public async Task PrintDataAsync()
{
await DownloadDataAsync();
console.WriteLine(“下载完成”);
}
3. 异常处理
使用 await 时,异常会被封装在 Task 中。需要用 try-catch 捕获:
public async Task SafeDownloadAsync()
{
try
{
string content = await HttpClient.GetStringAsync(“https://invalid-url”);
}
catch (HttpRequestException ex)
{
Console.WriteLine(“请求失败: ” + ex.Message);
}
}
注意:异常在 await 时抛出,而不是在调用异步方法时。
4. 并发执行多个任务
如果想同时运行多个异步操作并等待它们全部完成,可以使用 Task.WhenAll:
public async Task FetchMultipleSitesAsync()
{
var task1 = HttpClient.GetStringAsync(“https://site1.com”);
var task2 = HttpClient.GetStringAsync(“https://site2.com”);
var task3 = HttpClient.GetStringAsync(“https://site3.com”);
await Task.WhenAll(task1, task2, task3);
Console.WriteLine($”结果长度: {task1.Result.Length}, {task2.Result.Length}”);
}
这样三个请求是并发进行的,总耗时接近最长的那个请求。
如果想逐个处理结果,可以用 Task.WhenAny 实现“谁先完成就处理谁”。
5. 避免死锁
在某些场景下(如winForm或ASP.net旧版本),错误地调用异步方法可能导致死锁。例如:
// 危险!可能死锁
var result = DownloadDataAsync().Result;
应始终使用 await,而不是 .Result 或 .Wait()。如果必须在同步上下文中调用异步方法,应使用专门设计的模式(如 StartNew + Unwrap)或重构为全异步。
6. 实际应用场景
常见使用 async/await 的地方包括:
- Web API 调用(HttpClient)
- 文件读写(File.ReadAllTextAsync, streamWriter.WriteAsync)
- 数据库操作(EF Core 的 SaveChangesAsync)
- 定时器或延迟(Task.Delay)
- UI应用中防止界面卡顿
基本上就这些。掌握 async/await 的关键是理解“异步非阻塞”的本质,并合理使用 await 解决回调地狱问题。不复杂但容易忽略细节,比如返回类型和异常处理。