C# 异步陷阱(Task.Run)

本文探讨了C#中使用Task.Run将同步操作转换为异步操作时可能遇到的问题。尽管这可以避免UI线程阻塞,但对于性能并无实质性提升,尤其是对于IO密集型任务。解决方案是利用系统的Async函数实现真正的异步操作,避免专用线程等待,提高系统处理大量工作的能力。
摘要由CSDN通过智能技术生成

个人观点,仅供参考

C#为开发者提供了可以说最好的异步API了,只需要Task.Run即可将原本同步耗时的API转化为异步API,但其中也有个"巨坑"

Demo

看一个简单的例子

public async Task<string> MyTaskAsync(Uri address) 
{
    return await Task.Run(() =>
    {
        using (var client = new WebClient()) 
        {
            return client.DownloadString(address);
        }
    });
}
复制代码

乍一眼看这个函数似乎很正常,用一个Task包裹一个耗时的WebClient.DownloadString函数。但是我们需要问一下自己:

  1. WebClient.DownloadString 是CPU密集型,还是IO密集型
  2. 在Task.Run所在的托管线程里,是阻塞的还是非阻塞的

答案很明确:IO密集型,并且仍然是阻塞的,那么这意味着什么:我们只是换了一个地方阻塞,也就是类似于拆东墙补西墙,这样的代码确实可以使UI不再卡顿,但实际上对于性能的可拓展并没有什么好处。

Solve

解决方案很简单,就

`Task.Run` 是 C# 中的一个方法,用于在后台线程上异步启动一项工作,它属于 .NET Framework 的 `System.Threading.Tasks` 命名空间中的 `Task` 类。`Task.Run` 方法通常用于将 CPU 密集型的操作移至后台线程,以避免阻塞当前线程(例如,UI 线程)。该方法返回一个 `Task` 对象,该对象代表异步操作,可以用来处理异步操作的结果或者检查异步操作的完成状态。 要使用 `Task.Run`,你需要在代码文件顶部添加以下 using 指令: ```csharp using System.Threading.Tasks; ``` 然后,你可以像下面这样调用 `Task.Run` 来执行后台操作: ```csharp Task.Run(() => { // 这里放置需要在后台执行的代码 // 例如:CPU密集型计算任务 }); ``` 如果你需要等待后台任务完成并获取其结果,可以使用 `await` 关键字(这要求你的方法是异步的,即返回 `Task` 或 `Task<T>`): ```csharp public async Task MyMethodAsync() { await Task.Run(() => { // 执行后台操作 }); // 继续执行其他操作 } ``` 或者,如果你不需要等待后台任务完成,也可以使用 `Task.Run` 的返回值: ```csharp Task myTask = Task.Run(() => { // 执行后台操作 }); // 在其他地方可以检查 myTask 的状态或等待它完成 myTask.Wait(); // 如果你想要同步等待,但通常不推荐这样做 ``` 需要注意的是,`Task.Run` 适用于可并行化(CPU密集型)的操作,而对于 I/O 密集型操作,通常推荐使用 `Task.WhenAll` 或 `async` 和 `await` 关键字配合异步方法来实现非阻塞操作。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值