在 .NET Core 2.1 中 HttpClient 的 Bug 终于被修复了

自从用上 .NET Core 之后,网站的性能貌似有提升了,但经常“无缘无故”的将服务器资源耗尽,网站死掉。最开始,还以为是 Redis 碎片的原因,但是后来,一次偶然的机会,终于发现是框架内置的 HttpClient,错怪了 Redis。

在 C# 环境中,释放 HttpClient 资源,很多时候,我们都是使用的如下语句:

using (var client = new HttpClient())
{
    // 业务代码
}

但是,你知道?

using 语句在这里,并不好用。确切的说是:using 不能释放 HttpClient 所占用的资源。

同时,在 .NET Core 2.1 之前,如果要使用 HttpClient 去访问资源的,就连微软官方都建议写成:静态方法。但是,这样做,带来的问题显而易见,在不重启应用的前提下,HttpClient 中的 DNS 永远不会被刷新。
详情请看这里

现在,在 .NET Core 2.1 中,该 Bug 终于得到解决。

HttpClientFactory

记住(还是惯性思维),在 .NET Core 的世界里:万物皆 DI

步骤:

  1. 实现业务功能的 Client 端
  2. 在 Startup.cs 去注入(含调用出错时的重试)
  3. 在 PageModel 或 Controller 中去调用即可

首先,实现一个获取用户信息的 Client 端:

public class UserClient
{
    private readonly HttpClient _httpClient;

    public UserClient(HttpClient httpClient)
    {
        _httpClient = httpClient;
    }

    public async Task<string> GetUserByIdAsync(int id)
    {
        var result = await _httpClient.GetAsync($"/api/user/{id}");
        result.EnsureSuccessStatusCode();
        return await result.Content.ReadAsStringAsync();
    }
}

然后,在 Startup.cs 中,去进行 DI,若调用 API 失败,每隔 50 毫秒,重试一次,总共 3 次。

public void ConfigureServices(IServiceCollection services)
{
    // 其它业务代码

    services.AddHttpClient("userClient", client =>
    {
        // 配置用户 API 的主域名
        client.BaseAddress = new Uri("http://localhost:44361");
    })
    .ConfigurePrimaryHttpMessageHandler(() =>
    {
        return new HttpClientHandler()
        {
            // 若需要 GZip 解码,可在此配置
            AutomaticDecompression = DecompressionMethods.GZip | DecompressionMethods.Deflate
        };
    })
    .AddHttpMessageHandler(() => new ApiRetryHandler(3))   // 若调用失败,重试 3 次
    .AddTypedClient<UserClient>();

    // 其它业务代码
}

上面涉及到重试的 Handler,代码如下:

public class ApiRetryHandler : DelegatingHandler
{
    private readonly int _maxRetryCount;

    public ApiRetryHandler(int maxRetryCount = 5)
    {
        _maxRetryCount = maxRetryCount;
    }

    protected async override Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
    {
        HttpResponseMessage result = null;

        for (var i = 0; i < _maxRetryCount; i++)
        {
            try
            {
                result = await base.SendAsync(request, cancellationToken);
                result.EnsureSuccessStatusCode();
                return result;
            }
            catch (HttpRequestException) when (i == _maxRetryCount - 1)
            {
                throw;
            }
            catch (HttpRequestException)
            {
                await Task.Delay(TimeSpan.FromMilliseconds(50));   // 间隔 50 毫秒重试
            }
        }

        throw null;
    }
}

至此,HttpClient 的实现就完成一大半了,接下来,就是调用了,比如在 Index 页面中调用,代码如下:

public class IndexModel : PageModel
{
    private readonly UserClient _userCli;

    public IndexModel(UserClient userCli)
    {
        _userCli = userCli;
    }

    public async Task OnGetAsync()
    {
        var user = await _userCli.GetUserByIdAsync(1);
    }
}

这样,一个完整的 HttpClient 的调用就完成了。

写在最后

结合第三方库(比如:Polly),可更好的实现重试效果,推荐看看以下文章:

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,026评论 19 139
  • 1. ASP.NET Core中间件详解[#1-aspnet-core%E4%B8%AD%E9%97%B4%E4%...
    xdpie阅读 1,075评论 0 4
  • 走着走着, 心中戚戚作响。 经历种种未知, 聆听感恩, 方明珍惜。 宁静花开, 万物齐香。 剪一段流年,品味真谛。...
    狮子无问阅读 341评论 0 1
  • 今天听同事L说昨天她闺蜜因为老公家暴跑到她家,一身的伤痕累累,信誓旦旦要离婚。结果闺蜜老公找来,L把她老公骂得狗血...
    西海紫妖阅读 289评论 0 0
  • 问题描述给定一个年份,判断这一年是不是闰年。 当以下情况之一满足时,这一年是闰年: 年份是4的倍数而不是100的倍...
    FiveZM阅读 321评论 0 0