时间:2022-11-05 12:22:43 | 栏目:.NET代码 | 点击:次
async Task 语法糖出来后,异步编程变得非常简单,适合需要耗费较长时间的任务。
有些小伙伴使用后可能会非常疑惑,使用异步和同步,在耗时上几乎没有差别。
下面我们看一个例子,场景是需要调用多个第三方的WebApi,分别是获取名称、年龄、性别,由于网络环境等原因,api响应时间可能会接近1秒
public async Task Test() { var sw = new Stopwatch(); sw.Start(); var userName = await GetUserNameAsync(); var userAge = await GetUserAgeAsync(); var userSex = await GetUserSexAsync(); sw.Stop(); var ts = sw.Elapsed; Console.WriteLine($"总共耗时:{ts.TotalMilliseconds}ms"); } private async Task<string> GetUserNameAsync() { await Task.Delay(500); return "小明"; } private async Task<string> GetUserAgeAsync() { await Task.Delay(800); return "11"; } private async Task<string> GetUserSexAsync() { await Task.Delay(900); return "11"; }
运行后发现,这个时间2秒多,这用户体验肯定是无法忍受的
导致这样结果的原因是每次进行异步调用的时候,都在异步函数前加上了 await ,对于单单这个过程来说,其实相当于同步,等待直到结果返回,每个异步函数都await,时间自然就叠加了,为了解决这个问题,使用一个小技巧,可以将代码改成下面这样
public async Task Test() { var sw = new Stopwatch(); sw.Start(); var userNameTask = GetUserNameAsync(); var userAgeTask = GetUserAgeAsync(); var userSexTask = GetUserSexAsync(); var userName = await userNameTask; var userAge = await userAgeTask; var userSex = await userSexTask; sw.Stop(); var ts = sw.Elapsed; Console.WriteLine($"总共耗时:{ts.TotalMilliseconds}ms"); } private async Task<string> GetUserNameAsync() { await Task.Delay(500); return "小明"; } private async Task<string> GetUserAgeAsync() { await Task.Delay(800); return "11"; } private async Task<string> GetUserSexAsync() { await Task.Delay(900); return "11"; }
这次运行的总耗时,就是3个异步中,耗时最长那个GetUserSexAsync
为什么会这样呢,这个小技巧的关键是这里,当执行到异步函数的时候,不加 await,不进行等待,让这些任务乖乖在别的线程的执行,当需要用到他们的时候,再去等待返回值,所以时间上不会进行叠加,哪个最长,总耗时就是哪个
var userNameTask = GetUserNameAsync(); var userAgeTask = GetUserAgeAsync(); var userSexTask = GetUserSexAsync(); var userName = await userNameTask; var userAge = await userAgeTask; var userSex = await userSexTask;