RestTemplate未使用線程池問題的解決方法

一、問題描述

現場出現springboot服務卡死,無法打開頁面現象。

初步分析為服務中使用RestTemplate通信框架,但未使用連接池,如果通信拋出異常(連接失敗),連續運行一定時間,導致線程飆升,資源耗盡,服務程序宕機。

二、問題再現

模擬無法通信的微服務地址,修改連接2s/次,啟動三個微服務demo進行通信,連續測試2小時,現象可再現:

詳細如下圖:

啟動時線程數:

連接異常提示:

線程飆升:

大量未關閉線程:

線程dump信息:

“http-nio-8081-exec-120” #216 daemon prio=5 os_prio=0 tid=0x000000002b0f9800 nid=0x4a28 runnable [0x0000000030349000]
   java.lang.Thread.State: RUNNABLE
       at java.net.SocketInputStream.socketRead0(Native Method)
        at java.net.SocketInputStream.socketRead(SocketInputStream.java:116)
    …
       at org.springframework.http.client.AbstractClientHttpRequest.execute(AbstractClientHttpRequest.java:53)
        at org.springframework.web.client.RestTemplate.doExecute(RestTemplate.java:737)
        at org.springframework.web.client.RestTemplate.execute(RestTemplate.java:672)
        at org.springframework.web.client.RestTemplate.getForObject(RestTemplate.java:313)

三、問題分析:

主動健康檢查時,RestTemplate默認情況下不使用連接池,每次調用都會打開一個新的本地臨時端口和一個新連接,如果通信異常,會導致連接不被回收,持續通信,它會不斷新建線程,並且很快突破本地可用端口限制范圍,導致服務卡死。

四、解決方案:

使用RestTemplate連接池,設置ReadTimeout、ConnectTimeout超時時間,進行連接回收。

五、回歸驗證:

修改後,驗證如下:

初始線程:

測試3小時結束時線程:

線程池線程未增加,狀態交替

到此這篇關於RestTemplate未使用線程池問題的解決方法的文章就介紹到這瞭,更多相關RestTemplate未使用線程池內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: