自从用上 .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。
步骤:
- 实现业务功能的 Client 端
- 在 Startup.cs 去注入(含调用出错时的重试)
- 在 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),可更好的实现重试效果,推荐看看以下文章: