贝利信息

c# IHttpClientFactory 和 new HttpClient() 的区别

日期:2026-01-17 00:00 / 作者:幻夢星雲
直接 new HttpClient() 在高并发下崩,因频繁创建导致 TIME_WAIT 端口耗尽、连接池与 DNS 缓存不复用、配置分散且易泄漏;IHttpClientFactory 通过共享 SocketsHttpHandler、统一配置、自动生命周期管理解决。

直接 new HttpClient() 为什么会在高并发下崩?

不是它“不能用”,而是频繁 new HttpClient() 会快速耗尽本地端口(TIME_WAIT 状态堆积),尤其在短生命周期服务(如 ASP.NET Core Controller 每次请求都 new 一个)中,几分钟内就可能触发 SocketException: Only one usage of each socket address is permitted。根本原因在于:HttpClient 虽轻量,但底层的 SocketsHttpHandler 是重量级资源——它持有连接池、SSL 会话缓存、DNS 解析结果。每次 new 都新建一套,既不复用,也不释放干净。

IHttpClientFactory 怎么解决这些问题?

IHttpClientFactory 不是“造 HttpClient 的工厂”,而是“管理连接池 + 注入策略 + 统一配置”的协调者。它背后只维护一组共享的 SocketsHttpHandler 实例,所有通过它创建的 HttpClient 都复用同一套底层连接池和 DNS 缓存策略(默认每 2 分钟刷新一次 DNS)。

构造函数里该注入 IHttpClientFactory 还是 HttpClient?

必须注入 IHttpClientFactory,而不是直接注入 HttpClient。后者看似方便,实则是陷阱:

正确姿势是:

public class PaymentService
{
    private readonly IHttpClientFactory _factory;
    public PaymentService(IHttpClientFactory factory) => _factory = factory;

    public async Task ProcessAsync()
    {
        var client = _factory.CreateClient("PaymentApiClient"); // 名称匹配注册时的 key
        await client.PostAsync("/charge", content);
    }
}

命名客户端 vs 类型化客户端,怎么选?

两者本质都是为了解耦配置与使用,区别在于绑定方式:

注意:类型化客户端内部仍走工厂机制,不是 new 出来的;它的 HttpClient 参数由 DI 自动注入,生命周期受工厂管控——这点常被忽略。

真正容易被忽略的点是:即使用了 IHttpClientFactory,如果在非 DI 管理的类(比如静态工具类、BackgroundService 手动 new 的对象)里调用 CreateClient,依然可能因工厂未被正确释放或作用域错乱引发问题。工厂本身也依赖 DI 容器的生命周期管理,脱离上下文就失效。