HttpClient的一种错误使用方式
问题
在.Net相关的开发中经常需要使用到各种非托管或不受GC控制的资源,这类资源管理类大多实现了IDisposable接口。在使用完毕后,我们需要通过该接口及时清理它们。
微软的手册中建议我们使用using
关键字,该关键字结束后会隐式地调用Disposable进行清理,而不用我们手动调用Dispose
方法。
HttpClient是.Net提供的一个用于分配TCP连接资源,处理Http请求和响应的类。它在初始化时将申请一个套接字资源并创建连接,所以很自然地,我们会这样子使用它:
1 | using(var client = new HttpClient()) { |
接下来我们来模拟该操作执行多次的情景:
1 | // 使用using创建一个HttpClient实例来Get一些数据 |
我们创建了50个HttpClient实例,并且在每次using结束后都清理了其占用的所有资源。非常合理。
但如果我们在Shell中通过netstat
指令查看本机的连接状态会发现有大量的重复连接信息,并且它们都处于TIME_WAIT
状态(即该连接已经由某一方断开了连接)。
原因
实际上,HttpClient类的设计目的是仅实例化一次的单例对象,并在程序的整个生命周期内重复使用。HttpClient在初始化时将创建一个该实例专属的连接池,在调用Dispose释放HttpClient实例时,它占用的TCP端口并不会在连接关闭后立刻释放,而是会遵循TCP协议保持TIME-WAIT状态一段时间,该时间取决于系统设置。
解决方法
在第一次创建出HttpClient实例后,在非必要的情况下不要Dispose,重用这个实例
1 | private static HttpClient Client = new HttpClient(); |
运行这个例子,可以看到保持着的连接只剩下一个了,并且处于ESTABLISHED
激活状态
另一个问题
重用HttpClient实例会产生另一个问题:当DNS发生变更时无法重新加载和解析,必须重新创建实例。
在这种情形下,微软建议我们使用IHttpClientFactory
来管理HttpClient的生命周期,并执行各类操作