HttpClient api-连接池
【README】
本文 refer2 HttpClient Tutorialhttps://hc.apache.org/httpcomponents-client-4.5.x/current/tutorial/pdf/httpclient-tutorial.pdf
【2.3】http連接管理器
【2.3.1】可管理的連接與連接管理器
1)http連接簡(jiǎn)述
HTTP 連接是復(fù)雜的、有狀態(tài)的、線程不安全的對(duì)象,需要正確管理以功能正常。
HTTP 連接一次只能被一個(gè)執(zhí)行線程使用。
HttpClient 使用一個(gè)稱為 http連接管理器 的實(shí)體來(lái)管理對(duì) HTTP 連接的訪問(wèn)?,并由 HttpClientConnectionManager 接口表示。
HTTP 連接管理器的目的是作為新的 HTTP 連接工廠,管理持久連接的生命周期
并同步控制(synchronized-同步訪問(wèn))對(duì)持久連接的訪問(wèn)??,以確保一次只有一個(gè)線程訪問(wèn)一個(gè)連接。
2)http連接管理器
1.在內(nèi)部,http連接管理器?HttpClientConnectionManager實(shí)現(xiàn)類 與ManagedHttpClientConnection的多個(gè)實(shí)例一起作為實(shí)際連接的代理,以管理連接狀態(tài),控制IO操作執(zhí)行;
2.如果管理的連接被消費(fèi)者釋放或顯式關(guān)閉,底層連接會(huì)脫離它的代理并返回給http管理器;即使服務(wù)消費(fèi)者仍然持有對(duì)連接代理實(shí)例的引用,但它不能夠有意或無(wú)意執(zhí)行任何 I/O 操作或修改實(shí)際連接狀態(tài)。
3)從連接管理器獲取連接的示例代碼
HttpClientContext context = HttpClientContext.create(); HttpClientConnectionManager connMrg = new BasicHttpClientConnectionManager(); HttpRoute route = new HttpRoute(new HttpHost("localhost", 80)); // Request new connection. This can be a long process ConnectionRequest connRequest = connMrg.requestConnection(route, null); // Wait for connection up to 10 sec HttpClientConnection conn = connRequest.get(10, TimeUnit.SECONDS); try {// If not openif (!conn.isOpen()) {// establish connection based on its route infoconnMrg.connect(conn, route, 1000, context);// and mark it as route completeconnMrg.routeComplete(conn, route, context);}// Do useful things with the connection. } finally {connMrg.releaseConnection(conn, null, 1, TimeUnit.MINUTES); }?如有必要,可以通過(guò)調(diào)用 ConnectionRequest#cancel() 提前終止連接請(qǐng)求。 這將解除阻塞在 ConnectionRequest#get() 方法中的線程。
【2.5】http連接清理策略
1)問(wèn)題
經(jīng)典阻塞 I/O 模型的主要缺點(diǎn)之一是網(wǎng)絡(luò)套接字僅在IO操作阻塞時(shí)響應(yīng)I/O 事件。
當(dāng)一個(gè)連接被釋放回管理器時(shí),它可以保持活動(dòng)狀態(tài),但無(wú)法監(jiān)視套接字的狀態(tài)并對(duì)任何 I/O 事件做出反應(yīng)。
如果連接在服務(wù)器端關(guān)閉,則客戶端連接無(wú)法檢測(cè)到連接狀態(tài)的變化(通過(guò)關(guān)閉終端套接字做出適當(dāng)?shù)姆磻?yīng))。
2)解決方法
HttpClient 嘗試通過(guò)測(cè)試連接是否“過(guò)時(shí)”來(lái)解決該問(wèn)題,但不是很有效,因?yàn)樵谑褂眠B接執(zhí)行http請(qǐng)求之前,服務(wù)器連接被關(guān)閉了。陳舊的連接檢查不是 100% 可靠的。
要想讓空閑連接的socket模型不占用線程資源,唯一可行方法是使用專用監(jiān)控線程(如定時(shí)器線程)以清除因長(zhǎng)期不活躍的過(guò)期連接;
監(jiān)控線程可以定期調(diào)用 ClientConnectionManager#closeExpiredConnections()
關(guān)閉所有過(guò)期連接并連接池中清除。
還可以選擇性調(diào)用 ClientConnectionManager#closeIdleConnections() 方法來(lái)關(guān)閉在給定時(shí)間段內(nèi)所有空閑連接。
總結(jié)
以上是生活随笔為你收集整理的HttpClient api-连接池的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 新年快乐的英语怎么说 新年快乐的英语怎么
- 下一篇: 转:java网络编程-HTTP编程