考虑用Task.WhenAll

Posted dotNET跨平台

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了考虑用Task.WhenAll相关的知识,希望对你有一定的参考价值。

异步能在一定场景中带性能的飞跃,同步调用性能,也以带来时间的节省。

先看一下被调用的api:

using Microsoft.AspNetCore.Mvc;
using Microsoft.Extensions.Logging;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;


namespace WebAPI.Controllers
{
    [ApiController]
    [Route("[controller]")]
    public class HomeController : ControllerBase
    {
        private readonly ILogger<HomeController> _logger;
        public HomeController(ILogger<HomeController> logger)
        {
            _logger = logger;
        }
        [HttpGet("/api001")]
        public async Task<IActionResult> GetAPI001()
        {
            _logger.LogInformation("GetAPI001");
            await Task.Delay(1000);
            return new JsonResult(new { result = true, data = "api001 返回成功" });
        }
        [HttpGet("/api002")]
        public async Task<IActionResult> GetAPI002()
        {
            _logger.LogInformation("GetAPI002");
            await Task.Delay(1000);
            if (DateTime.Now.Second % 2 == 0)
            {
                throw new Exception("api002异常");
            }
            return new JsonResult(new { result = true, data = "api002 返回成功" });
        }
        [HttpGet("/api003")]
        public async Task<IActionResult> GetAPI003()
        {
            _logger.LogInformation("GetAPI003");
            await Task.Delay(1000);
            return new JsonResult(new { result = true, data = "api003 返回成功" });
        }
    }
}

调用时反序列化的实体类

    class ResponseResult<T>
    {
        public bool Result { get; set; }
        public string Message { get; set; }
        public T Data { get; set; }
    }

三个api的调用方法

private static async Task<string> GetAPI001(HttpClient httpClient)
{
    var content = await httpClient.GetStringAsync("http://localhost:5000/api001");
    var result = JsonSerializer.Deserialize<ResponseResult<string>>(content, new JsonSerializerOptions { PropertyNameCaseInsensitive = true });
    if (result.Result)
    {
        return result.Data;
    }
    else
    {
        return result.Message;
    }
}
private static async Task<string> GetAPI002(HttpClient httpClient)
{
    var content = await httpClient.GetStringAsync("http://localhost:5000/api002");
    var result = JsonSerializer.Deserialize<ResponseResult<string>>(content, new JsonSerializerOptions { PropertyNameCaseInsensitive = true });
    if (result.Result)
    {
        return result.Data;
    }
    else
    {
        return result.Message;
    }
}
private static async Task<string> GetAPI003(HttpClient httpClient)
{
    var content = await httpClient.GetStringAsync("http://localhost:5000/api003");
    var result = JsonSerializer.Deserialize<ResponseResult<string>>(content, new JsonSerializerOptions { PropertyNameCaseInsensitive = true });
    if (result.Result)
    {
        return result.Data;
    }
    else
    {
        return result.Message;
    }
}


同步的调用方式

static async Task SyncCall()
{
    using var httpClient = new HttpClient();
    try
    {
        var result1 = await GetAPI001(httpClient);
        WriteLine(result1);
    }
    catch (Exception exc)
    {
        WriteLine(exc.Message);
    }
    try
    {
        var result2 = await GetAPI002(httpClient);
        Console.WriteLine(result2);


    }
    catch (Exception exc)
    {
        WriteLine(exc.Message);
    }
    try
    {
        var result3 = await GetAPI003(httpClient);
        Console.WriteLine(result3);
    }
    catch (Exception exc)
    {
        WriteLine(exc.Message);
    }
}

调用方式

static async Task Main(string[] args)
{
    while (true)
    {
        WriteLine("回车开始执行");
        ReadLine();
        var stopwatch = Stopwatch.StartNew();
        await SyncCall();
        WriteLine($"用时{stopwatch.ElapsedMilliseconds}ms");
    }
}

同步的调用,运行三次调用是三次的时间,3202ms,如果有异常不干扰其他api调用。

static async Task AsyncCall()
{
    using var httpClient = new HttpClient();
    var allTasks = Task.WhenAll(GetAPI001(httpClient), GetAPI002(httpClient), GetAPI003(httpClient));
    try
    {
        var results = await allTasks;
        foreach (var result in results)
        {
            Console.WriteLine(result);
        }
    }
    catch (Exception exc)
    {
        Console.WriteLine($"捕捉到的异常:{exc.Message}");
    }
    if (allTasks.Exception != null)
    {
        Console.WriteLine($"AllTasks异常:{ allTasks.Exception.Message}");
    }
}


同步调用成功时间是1156ms,时间缩短了,但三个api调用,如果有异常,则全军覆没。

先择适合的方式,打造更优的应用。

以上是关于考虑用Task.WhenAll的主要内容,如果未能解决你的问题,请参考以下文章

为啥不等待 Task.WhenAll 抛出 AggregateException?

可以使用 Task.WhenAll(...) 或其他比每次等待更有意义的东西重写此异步/等待代码吗? [复制]

使用 Task.WhenAll 一次向我的 WebAPI 发送多个请求

Task.WhenAll 不等待任务完成[关闭]

csharp Task.WhenAllを利用している场合のタイムアウト处理の书き方(Task.WhenAll,Task.WhenAny,Task.Delay)

Task.WhenAll(taskList).Wait() 是不是与 Task.WaitAll(taskList) 相同?