在 .NET Core 2.1 中 HttpClient 的 B

2018-06-07  本文已影响0人  缺水的海豚

自从用上 .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),可更好的实现重试效果,推荐看看以下文章:

上一篇 下一篇

猜你喜欢

热点阅读