常见的 async/await 误用

本文总结了使用async/await时可能导致性能下降的常见错误,包括不必要的异步修饰符、长时间运行操作、异步void方法、未await的using块以及嵌套Task的不当使用。通过修正这些错误并引入AsyncFixer工具,可以提升异步代码的效率和可维护性。

平时使用async/await 时,本来想着提升一下吞吐量,但莫名发现性能下降的厉害,所以总结了一下正确的使用方式,避免入坑。

1.使用不必要的 async/await

有些方法不需要使用async/await。添加异步修饰符是有代价的:编译器将在每个异步方法中生成一些代码。
下列代码开启了一个外部 Task,并不需要等待完成:

//修改前
public static async Task Demo()
{
    await Task.Factory.StartNew(() => Console.WriteLine("My IO"));
}

//修改后
public static Task Demo()
{
    return Task.Factory.StartNew(() => Console.WriteLine("My IO"));
}

2.异步方法内的长时间运行或阻塞操作

在异步方法中使用一些可能长时间运行或阻塞的操作,即使有这些方法的相应异步版本。
下列代码使用了ReadToEnd,而不是对应的异步版本:

//修改前
public static async Task Demo()
{
    StreamReader reader = GetReader();

    var str = reader.ReadToEnd();
}

//修改后
public static async Task Demo()
{
    StreamReader reader = GetReader();

    var str = await reader.ReadToEndAsync();
}

3.异步 void 方法

异步 void 方法中的异常无法在调用方法中捕获。
下列代码运行时并不会抛出异常:

//修改前
public static async void Demo()
{
    throw new Exception();
    await Task.Delay(300);
    await Task.Delay(300);
}

//修改后
public static async Task Demo()
{
    throw new Exception();
    await Task.Delay(300);
    await Task.Delay(300);
}

4.在 using 块中未 await

在 using 块内,执行了异步方法但未 await,它可能导致潜在的异常或错误的结果。
下列代码中的CopyToAsync如果持续时间很长,将抛出 ObjectDisposedException:

//修改前
using (var stream = new FileStream("newfile.txt", FileMode.Open))
{
    newStream.CopyToAsync(stream);
}

//修改后
using (var stream = new FileStream("newfile.txt", FileMode.Open))
{
    await newStream.CopyToAsync(stream);
}

5.从嵌套 Task 转换为外部 Task

这通常发生在将 async/await 关键字与 Task.Factory.StartNew 混用的情况,没有办法等待并获得子任务的结果。
下列代码中的意图是进行一秒的延迟,但实际不会有任何延迟:

//修改前
public static async Task Demo()
{
    Console.WriteLine("Hello");
    await Task.Factory.StartNew(() => Task.Delay(1000));
    Console.WriteLine("My IO");
}

//修改后
public static async Task Demo()
{
    Console.WriteLine("Hello");
    await Task.Run(() => Task.Delay(1000));
    Console.WriteLine("My IO");
}

辅助工具

但是要完全靠人去识别这些错误有点难度,这里介绍一个查找和纠正 async/await 的常见误用的工具 —— AsyncFixer。
引用 Nuget 包 AsyncFixer 后, 在 VS 中就会将这些误用视为警告,也可以将严重性调为“错误":
(找到对应的分析器,设置“严重性”为“错误”)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值