Task提高异步执行效率技巧

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 GetUserNameAsync()
{
    await Task.Delay(500);
    return "小明";
}

private async Task GetUserAgeAsync()
{
    await Task.Delay(800);
    return "11";
}

private async Task 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 GetUserNameAsync()
{
    await Task.Delay(500);
    return "小明";
}

private async Task GetUserAgeAsync()
{
    await Task.Delay(800);
    return "11";
}

private async Task 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;

到此这篇关于Task提高异步执行效率技巧的文章就介绍到这了。希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

你可能感兴趣的:(Task提高异步执行效率技巧)