最近挺多童鞋問我如何配置Spring Cloud xxx元件的重試。本篇進行一個總結。
Spring Cloud中的重試機制應該說是比較混亂的,不同的版本有一定差別,實作也不大一樣,好在Spring Cloud Camden之後已經基本穩定下來,Dalston中又進行了一些改進,詳情暫且不表。
下面我們來詳細探讨。
筆者使用的版本是
Spring Cloud Dalston SR4
,同樣适應于 Edgware
以及更高版本,對于 Dalston
此前的版本,本文不做讨論,大家可自行研究。 Ribbon+RestTemplate的重試
對于整合了Ribbon的RestTemplate,例如一個RestTemplate添加了
@LoadBalanced
注解:
@Bean
@LoadBalanced
public RestTemplate restTemplate() {
SimpleClientHttpRequestFactory simpleClientHttpRequestFactory = new SimpleClientHttpRequestFactory();
simpleClientHttpRequestFactory.setConnectTimeout(1000);
simpleClientHttpRequestFactory.setReadTimeout(1000);
return new RestTemplate(simpleClientHttpRequestFactory);
}
在此基礎上,使用如下配置,即可實作重試:
spring:
cloud:
loadbalancer:
retry:
enabled: true
ribbon:
# 同一執行個體最大重試次數,不包括首次調用
MaxAutoRetries: 1
# 重試其他執行個體的最大重試次數,不包括首次所選的server
MaxAutoRetriesNextServer: 2
# 是否所有操作都進行重試
OkToRetryOnAllOperations: false
Feign的重試
Feign本身也具備重試能力,在早期的Spring Cloud中,Feign使用的是
feign.Retryer.Default#Default()
,重試5次。但Feign整合了Ribbon,Ribbon也有重試的能力,此時,就可能會導緻行為的混亂。
Spring Cloud意識到了此問題,是以做了改進,将Feign的重試改為
feign.Retryer#NEVER_RETRY
,如需使用Feign的重試,隻需使用Ribbon的重試配置即可。是以,對于Camden以及以後的版本,Feign的重試可使用如下屬性進行配置:
ribbon:
MaxAutoRetries: 1
MaxAutoRetriesNextServer: 2
OkToRetryOnAllOperations: false
相關Issue可參考:
https://github.com/spring-cloud/spring-cloud-netflix/issues/467Zuul的重試
配置:
zuul:
# 開啟Zuul的重試
retryable: true
ribbon:
MaxAutoRetries: 1
MaxAutoRetriesNextServer: 2
OkToRetryOnAllOperations: false
上面我們使用
zuul.retryable=true
對Zuul全局開啟了重試,事實上,也可對指定路由開啟/關閉重試:
zuul.routes.<routename>.retryable=true
局部配置優先級更高。
基于HTTP響應碼重試
clientName:
ribbon:
retryableStatusCodes: 404,502
注意點
- Hystrix的逾時時間必須大于逾時的時間,否則,一旦Hystrix逾時,就沒辦法繼續重試了。
- 一般來說,不建議将
設為true。因為一旦啟用該配置,則表示重試任何操作,包括POST請求,而由于緩存了請求體,此時可能會影響伺服器的資源。ribbon.OkToRetryOnAllOperations