成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

HttpClient 與 Close_Wait

網絡 網絡管理
服務器A需要通過HttpClient去連接另一個系統B提供的服務,運行一段時間后拋出以下異常:java.net.SocketException: Connection reset by peer: socket write error close_wait。

服務器A需要通過HttpClient去連接另一個系統B提供的服務,運行一段時間后拋出以下異常:java.net.SocketException: Connection reset by peer: socket write error close_wait

在服務器B上運行netstat命令,發現大量連接處于CLOSE_WAIT 狀態。

問題分析:

簡單來說CLOSE_WAIT數目過大是由于被動關閉連接處理不當導致的。

我說一個場景,服務器A會去請求服務器B上面的apache獲取文件資源,正常情況下,如果請求成功,那么在抓取完資源后服務器A會主動發出關閉連接的請求,這個時候就是主動關閉連接,連接狀態我們可以看到是TIME_WAIT。如果一旦發生異常呢?假設請求的資源服務器B上并不存在,那么這個時候就會由服務器B發出關閉連接的請求,服務器A就是被動的關閉了連接,如果服務器A被動關閉連接之后自己并沒有釋放連接,那就會造成CLOSE_WAIT的狀態了。

所以很明顯,問題還是處在程序里頭。

原始代碼塊:

  1. try 
  2. client = HttpConnectionManager.getHttpClient(); 
  3. HttpGet get = new HttpGet(); 
  4. get.setURI(new URI(urlPath)); 
  5. HttpResponse response = client.execute(get); 
  6. if (response.getStatusLine ().getStatusCode () != 200) { 
  7. return null; 
  8. HttpEntity entity =response.getEntity(); 
  9.  
  10. if( entity != null ){ 
  11. in = entity.getContent(); 
  12. ..... 
  13. return sb.toString (); 
  14.  
  15. catch (Exception e) 
  16. e.printStackTrace (); 
  17. return null; 
  18. finally 
  19. if (isr != null){ 
  20. try 
  21. isr.close (); 
  22. catch (IOException e) 
  23. e.printStackTrace (); 
  24. if (in != null){ 
  25. try 
  26. <span style="color:#ff0000;">in.close ();</span> 
  27. catch (IOException e) 
  28. e.printStackTrace (); 

HttpClient使用我們常用的InputStream.close()來確認連接關閉,分析上面的代碼,一旦出現非200的連接,這個連接將永遠僵死在連接池里頭,因為inputStream得不到初始化,永遠不會調用close()方法了。

 

通過代碼稍微修改,更嚴謹的處理異常情況就可以解決問題了:

  1. public static String readNet (String urlPath) 
  2. StringBuffer sb = new StringBuffer (); 
  3. HttpClient client = null
  4. InputStream in = null
  5. InputStreamReader isr = null
  6. HttpGet get = new HttpGet(); 
  7. try 
  8. client = HttpConnectionManager.getHttpClient(); 
  9. get.setURI(new URI(urlPath)); 
  10. HttpResponse response = client.execute(get); 
  11. if (response.getStatusLine ().getStatusCode () != 200) { 
  12. get.abort(); 
  13. return null; 
  14. HttpEntity entity =response.getEntity(); 
  15.  
  16. if( entity != null ){ 
  17. in = entity.getContent(); 
  18. ...... 
  19. return sb.toString (); 
  20.  
  21. catch (Exception e) 
  22. get.abort(); 
  23. e.printStackTrace (); 
  24. return null; 
  25. finally 
  26. if (isr != null){ 
  27. try 
  28. isr.close (); 
  29. catch (IOException e) 
  30. e.printStackTrace (); 
  31. if (in != null){ 
  32. try 
  33. in.close (); 
  34. catch (IOException e) 
  35. e.printStackTrace (); 

顯示調用HttpGet的abort,這樣就會直接中止這次連接,我們在遇到異常的時候應該顯示調用,因為誰能保證異常是在InputStream in賦值之后才拋出的呢。

more:

首先我們知道,如果我們的服務器程序處于CLOSE_WAIT狀態的話,說明套接字是被動關閉的!

因為如果是CLIENT端主動斷掉當前連接的話,那么雙方關閉這個TCP連接共需要四個packet:

 

Client –-> FIN  –-> Server
Client <–- ACK  <–- Server
這時候Client端處于FIN_WAIT_2狀態;而Server 程序處于CLOSE_WAIT狀態。
Client <–- FIN  <–- Server
這時Server 發送FIN給Client,Server 就置為LAST_ACK狀態。
Client –-> ACK  –-> Server
Client回應了ACK,那么Server 的套接字才會真正置為CLOSED狀態。

 

Server 程序處于CLOSE_WAIT狀態,而不是LAST_ACK狀態,說明還沒有發FIN給Client,那么可能是在關閉連接之前還有許多數據要發送或者其他事要做,導致沒有發這個FIN packet。

通常來說,一個CLOSE_WAIT會維持至少2個小時的時間(這個時間外網服務器通常會做調整,要不然太危險了)。如果有個流氓特地寫了個程序,給你造成一堆的CLOSE_WAIT,消耗

你的資源,那么通常是等不到釋放那一刻,系統就已經解決崩潰了。

只能通過修改一下TCP/IP的參數,來縮短這個時間:修改tcp_keepalive_*系列參數有助于解決這個問題。

但是實際上,還是主要是因為我們的程序代碼有問題,

 

more:

最近做httpclient做轉發服務,發現服務器上總是有很多close_wait狀態的連接,而且這些連接都不會關閉,最后導致服務器沒法建立新的網絡連接,從而停止響應。

后來在網上搜索了一下,發現解決的方法也很簡單,如果想重用連接,那就使用連接管理器,從連接管理器里獲取連接,然后定時的用連接管理器來釋放空閑連接。httpclient自帶了SimpleHttpConnectionManager,提供了

 

Java代碼 

  1. closeIdleConnections(long idleTimeout)  

這樣的方法。

如果不需要重用鏈接,則直接在httpmethod創建時,設置一個http頭信息就可以了

 

Java代碼

  1. httpmethod.setRequestHeader("Connection", "close");   

這樣就不會有惱人的close_wait了。

 

責任編輯:林琳
相關推薦

2015-03-27 10:18:25

TCP協議CLOSE_WAIT狀服務器異常

2011-08-25 14:00:48

close中文man

2011-08-24 10:24:26

CLOSE中文man

2024-08-20 08:05:14

WinformWndProc?Windows

2011-08-25 16:57:51

iconv_close中文man

2025-02-05 08:02:31

Winform開發窗體

2021-03-05 16:08:58

RestAPINET工具包

2011-05-26 15:52:31

sleep()wait()

2021-07-15 14:27:47

LinuxSocketClose

2014-08-08 09:39:40

httpClient

2021-06-06 13:07:06

.NETWindowsLinux

2009-06-16 13:22:59

SqlConnecti

2021-06-05 23:41:47

NET異常 HttpClient

2024-10-07 08:59:47

sleepwait線程

2024-08-29 09:11:38

GolangEOF連接池

2022-01-09 17:38:32

goHttpClient 集群

2024-09-13 09:32:30

2021-07-12 07:59:05

對象接口編程

2021-08-02 14:54:50

鴻蒙HarmonyOS應用

2020-03-24 15:15:29

HttpClientOkHttpJava
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲三区视频 | 水蜜桃亚洲一二三四在线 | 香蕉婷婷| 国产精品一区在线观看 | 亚洲第一av| 精品久久久久久久久久久久久久久久久 | 欧美jizzhd精品欧美巨大免费 | 亚洲精品一区二三区不卡 | 精品国产乱码一区二区三区a | 一区二区三区四区在线免费观看 | 成人午夜精品一区二区三区 | 亚洲日本欧美日韩高观看 | 色视频网站免费 | 国产精品永久在线观看 | 福利视频一区二区 | 国产精品久久久久aaaa樱花 | 午夜视频在线免费观看 | 2019天天干夜夜操 | 亚洲网站在线观看 | 久久成人国产精品 | 色综合久久伊人 | 成人美女免费网站视频 | 亚洲aⅴ| 久久久久亚洲精品 | 另类一区| 久久午夜精品福利一区二区 | 五月综合激情网 | 亚洲视频二区 | 三级黄色片在线 | 狠狠婷婷综合久久久久久妖精 | 欧美在线观看一区 | 国产不卡在线观看 | 日韩欧美一区二区三区免费观看 | 日韩一区二区在线观看视频 | 国产精品成人一区二区三区 | 亚洲一区二区三区免费视频 | 久久天天躁狠狠躁夜夜躁2014 | 午夜三级在线观看 | 伊人伊成久久人综合网站 | 国产高清美女一级a毛片久久w | 国产资源视频 |