视频1 视频21 视频41 视频61 视频文章1 视频文章21 视频文章41 视频文章61 推荐1 推荐3 推荐5 推荐7 推荐9 推荐11 推荐13 推荐15 推荐17 推荐19 推荐21 推荐23 推荐25 推荐27 推荐29 推荐31 推荐33 推荐35 推荐37 推荐39 推荐41 推荐43 推荐45 推荐47 推荐49 关键词1 关键词101 关键词201 关键词301 关键词401 关键词501 关键词601 关键词701 关键词801 关键词901 关键词1001 关键词1101 关键词1201 关键词1301 关键词1401 关键词1501 关键词1601 关键词1701 关键词1801 关键词1901 视频扩展1 视频扩展6 视频扩展11 视频扩展16 文章1 文章201 文章401 文章601 文章801 文章1001 资讯1 资讯501 资讯1001 资讯1501 标签1 标签501 标签1001 关键词1 关键词501 关键词1001 关键词1501 专题2001
.NET Core中HttpClient的正确打开方式
2020-11-27 22:34:42 责编:小采
文档

可是,当我们试图运行下面的测试:

public async Task SendRequest() 
{
 Console.WriteLine("Starting reqeust");
 for(int i = 0; i<10; i++)
 {
 using(var client = new HttpClient())
 {
 var result = await client.GetAsync("http://www.baidu.com");
 Console.WriteLine(result.StatusCode);
 }
 }
 Console.WriteLine("Reqeust done");
}

此时在terminal下列出所有端口:

netstat -ap tcp | grep -i "time_wait"

你会发现本地开启了10个端口,这说明HttpClient的工作原理其实跟我们认为的IDisposable是有区别的,如果你把HttpClient用作大规模的Http请求,实际上会创建很多个Http连接,而且这些资源并不能被立即释放。一个显而易见的改进方法是共享同一个HttpClient实例,从而达到节约socket资源的目的。

private static readonly HttpClient _client = new HttpClient();
public async Task SendRequest2() 
{
 _testOutputHelper.WriteLine("Start request");
 
 for(int i = 0; i<10; i++)
 {
 var result = await _client.GetAsync("http://www.baidu.com");
 Console.WriteLine(result.StatusCode);
 }
 _testOutputHelper.WriteLine("Request done");
}

这个方案似乎解决了问题,使用单例的HttpClient的确会减少Socket资源,但是这个方案会引发新的问题:由于这个Http连接始终保持连接状态,所以当请求地址的DNS发生更新的时候并不会应用到这个Http连接上。这个问题在微服务,高可用时代更加常见Singeton HttpClient doesn't respect DNS changes。

最终,一个叫做HttpClientFactory的开源实现用来彻底解决这个问题。微软也将HttpClientFactory集成在了.NET Core中。

在.NET Core中创建HttpClient
1.添加Nuget

Microsoft.Extensions.Http 

2.在Dependency Injection容器中注册服务

services.AddHttpClient();

3. 使用构造器注入使用IhttpClientFactory

public class BasicUsage
{
 private readonly IHttpClientFactory _clientFactory;

 public BasicUsage(IHttpClientFactory clientFactory)
 {
 _clientFactory = clientFactory;
 }

 public async Task SendRequest()
 {
 var request = new HttpRequestMessage(HttpMethod.Get, 
 "http://www.baidu.com");

 var client = _clientFactory.CreateClient();
 var response = await client.SendAsync(request);
 //do something for response
 }
}

4. 使用Named HttpClient

由于我们在DI容器中注册了唯一的HttpClientFactory,意味着通过HttpClientFactory创建出来的HttpClient可能是同一个配置和参数,如果你需要不同配置的HttpClient,你可以通过“起名字的”的方式注册不同的HttpClient。

services.AddHttpClient("baidu", c =>
{
 c.BaseAddress = new Uri("https://www.baidu.com");
 c.DefaultRequestHeaders.Add("Accept", "application/json");
});

一旦注册了一个名叫“baidu"的HttpClient,你就可以通过下面的方式来建创建HttpClient:

var client = _clientFactory.CreateClient("baidu");

5.集成Polly

Polly是一个用来故障处理库,它允许开发者在Http请求中添加“重试、熔断器、超时等”策略。

先添加Nuget:

Microsoft.Extensions.Http.Polly 

添加策略:

var timeout = Policy.TimeoutAsync<HttpResponseMessage>(TimeSpan.FromSeconds(10));

services.AddHttpClient("baidu")
 .AddPolicyHandler(request => timeout)
 .AddTransientHttpErrorPolicy(p=>p.RetryAsync(3));

当然还有一些高阶用法,详见Initiate HTTP requests,总之HttpClientFactory提供了一种高效实用HttpClient的方式,如果你还在自己new HttpClient,请赶快试试新的方案吧。

总结

下载本文
显示全文
专题