ASP.Net Core 是一个开源的,跨平台的,轻量级模块化框架,可用它来构建高性能的Web程序,这篇文章我们将会讨论如何在 ASP.Net Core 中使用 HttpClientFactory。
为什么要使用 HttpClientFactory
可以用 HttpClientFactory 来集中化管理 HttpClient,工厂提供了对 HttpClient 的创建,配置和调度,值得一提的是:HttpClient 一直都是 Http 请求业务方面的一等公民。
HttpClient 虽好,但它有一些缺点:
- 创建太多的 HttpClient 是一种低效的行为,因为当一个新客户端连接到远程 Server 时,你的应用程序还需要承担着重连远程 Server 的开销。 
- 如果每一个 request 都创建一个 HttpClient,当应用程序负载过大, Socket 必将耗尽,比如默认情况下 HttpClient 会维持至少4分钟的 Connection 连接。 
所以推荐的做法是创建一个可供复用的共享式 HttpClient 实例,如果你要打破沙锅问到低的话,即使是创建共享式的 HttpClient 也会有很多问题,比如它会无视 DNS 缓存生效,那怎么办呢?可以用 .NET Core 2.1 引入的 HttpClientFactory 来解决此问题。。。用它来统一化的高效管理 HttpClient。
使用 HttpClientFactory
HttpClientFactory 有两种使用方式。
- NamedClient 
- TypedClient 
所谓的 NamedClient 就是注册带有标记的 HttpClient 到 HttpClientFactory 工厂中,下面的代码展示了一个名为 IDGCustomApi 的 HttpClient 的工厂注册。
// This method gets called by the runtime. Use this method to add services to the container.public void ConfigureServices(IServiceCollection services){services.AddHttpClient("IDGCustomApi", client =>{client.BaseAddress = new Uri("https://localhost:6045/");client.DefaultRequestHeaders.Add("Accept", "application/json");client.DefaultRequestHeaders.Add("User-Agent", "IDG");});services.AddControllers();}所谓的 TypedClient 就是注册一个你自定义的 HttpClient,我想你肯定有点懵逼了,没关系,我现在就来自定义 HttpClient, 然后通过 AddHttpClient() 注册到容器中。
public class CustomHttpClient{public HttpClient Client { get; }public CustomHttpClient(HttpClient client){Client = client;}}public class Startup{// This method gets called by the runtime. Use this method to add services to the container.public void ConfigureServices(IServiceCollection services){services.AddHttpClient<CustomHttpClient>(client => client.BaseAddress = new Uri("https://localhost:6045/"));services.AddControllers();}}注入 Controller
为了能够在 Controller 中使用,可以将 IHttpClientFactory 通过构造函数方式进行注入,参考如下代码:
[ApiController][Route("[controller]")]public class WeatherForecastController : ControllerBase{private IHttpClientFactory httpClientFactory;public WeatherForecastController(ILogger<WeatherForecastController> logger, IHttpClientFactory httpClientFactory){this.httpClientFactory = httpClientFactory;}[HttpGet]public async Task<string> Get(){var httpClient = httpClientFactory.CreateClient("IDGCustomApi");string html = await httpClient.GetStringAsync("http://bing.com");return html;}}
从 IHttpClientFactory 的默认实现 DefaultHttpClientFactory 的源码也可以看出,httpClient 所关联的 HttpMessageHandler 和 Options 都被工厂跟踪和管控。
internal class DefaultHttpClientFactory : IHttpClientFactory, IHttpMessageHandlerFactory{public HttpClient CreateClient(string name){if (name == null){throw new ArgumentNullException("name");}HttpMessageHandler handler = CreateHandler(name);HttpClient httpClient = new HttpClient(handler, disposeHandler: false);HttpClientFactoryOptions httpClientFactoryOptions = _optionsMonitor.Get(name);for (int i = 0; i < httpClientFactoryOptions.HttpClientActions.Count; i++){httpClientFactoryOptions.HttpClientActions[i](httpClient);}return httpClient;}public HttpMessageHandler CreateHandler(string name){if (name == null){throw new ArgumentNullException("name");}ActiveHandlerTrackingEntry value = _activeHandlers.GetOrAdd(name, _entryFactory).Value;StartHandlerEntryTimer(value);return value.Handler;}}译文链接:https://www.infoworld.com/article/3276007/how-to-work-with-httpclientfactory-in-aspnet-core.html