前言
不知道你有沒有過這樣的經歷,你的Spring項目通過http接口遠程調用外部系統時,通常你會設置超時時間,比如5秒鐘,但是實際情況卻是由于外部系統出現故障并導致連接超時,有些請求花費遠遠超過5秒時間,甚至1分鐘,這直接導致你本身的接口響應很慢,如果訪問你接口的流量很大的話,甚至拖垮你的系統,這將會是災難性的后果。
你可能會好奇到底是什么原因導致的,很可能是你忽略了這個超時配置項connectionRequestTimeout導致的。
Spring中的連接池
在深入了解 connectionRequestTimeout 之前,我們需要了解外部 HTTP 請求是如何在 Spring 中進行的。
Spring 通過Http調用外部系統的時候,會使用連接池去管理他們。因為HTTP請求創建連接代價比較高,而連接池可以做到連接的復用,回過頭,我們思考下為什么會代價高呢?
- DNS 解析:在建立連接之前,客戶端必須首先使用域名系統 (DNS) 將服務器的域名解析為 IP 地址。DNS 解析可能需要時間,尤其是在客戶端的 DNS 緩存很冷且服務器的域名尚未緩存的情況下。
- TCP 握手:確定服務器的 IP 地址后,客戶端必須與服務器建立 TCP 連接。這涉及三次握手過程,這可能會花費時間并增加請求的開銷。
- SSL/TLS 協商:如果服務器使用 HTTPS,客戶端還必須在可以交換任何數據之前與服務器協商 SSL/TLS 連接。這涉及一個復雜的握手過程,可能會增加請求的大量開銷。
為 HTTP 請求使用連接池可以顯著提高性能。但是,我們需要對其進行適當配置以防止出現災難情況。
什么是連接請求超時?
現在讓我們用連接池的概念再來看問題。
本例中B組件是外部系統,處于無法建立HTTP連接的情況,導致A中的連接超時,請仔細看上面A指向連接池的箭頭,A 正在等待另一個連接來建立到 B 的 HTTP 連接。
connectionRequestTimeout 是 Spring 中的一個配置參數,用于確定客戶端在超時前等待來自連接池的連接的時間。此超時值用于防止客戶端無限期地等待可能不可用的連接,并在不再需要時釋放連接池中的資源。
API 響應時間 = connectionRequestTimeout? + connectionTimeout? + readTimeout 。
- connectionRequestTimeout:等待從連接池獲取連接的時間。
- connectionTimeout:等待與外部組件建立連接的時間。
- readTimeout:等待外部組件響應的時間。
如何設置connectionRequestTimeout?
connectionRequestTimeout 的默認值為 -1,這意味著它無限期地等待來自連接池的連接。由于我們希望避免外部組件中斷導致系統故障,因此我們需要為其設置一個顯式值。
假設您將值設置得太短:1 秒。如果系統需要高延遲,這可能是一個頻繁的故障,因為它可能一直渴望連接。另一方面,如果該值太長,比如 10 分鐘,系統很容易因外部故障而失敗。
因此,你需要統計系統的API響應時間, 最大超時響應時間如下:
API 響應時間 = connectionRequestTimeout + connectionTimeout + readTimeout。
看看是否有很多請求超過了connectionTimeout和readTimeout?之和, 如果是這樣的話,系統需要增大連接池的大小或者減小connectionRequestTimeout?值。否則就將 connectionRequestTimeout 合理設置為 15 秒到 30 秒之間。
下面是設置 connectionRequestTimeout 的示例代碼。
import org.springframework.http.client.HttpComponentsClientHttpRequestFactory;
import org.springframework.web.client.RestTemplate;
public class ConnectionRequestTimeoutExample {
public static void main(String[] args) {
RestTemplate restTemplate = new RestTemplate();
HttpComponentsClientHttpRequestFactory requestFactory = new HttpComponentsClientHttpRequestFactory();
// Set the connectionRequestTimeout value to 10 seconds
requestFactory.setConnectionRequestTimeout(10000);
restTemplate.setRequestFactory(requestFactory);
...
}
}
在這個例子中,我們創建了一個新的RestTemplate?和一個新的HttpComponentsClientHttpRequestFactory?,然后我們connectionRequestTimeout?使用該方法將該值設置為 10 秒,并使用該方法為 RestTemplate? 設置請求工廠setRequestFactory()。
總結
最后,我們在總結以下Spring中Http請求的3個關鍵的超時配置吧,其中connectionRequestTimeout 最容易被忽視的。
- connectionRequestTimeout:等待從連接池獲取連接的時間
- connectionTimeout:等待與外部組件建立連接的時間
- readTimeout:等待外部組件響應的時間