Feign Client 超時時間配置不生效
解決方案
Feign Client 的 connectTimeout 和 readTimeout 需要同時配置:
1
2
|
feign.client.config.my-api.connectTimeout = 10000 feign.client.config.my-api.readTimeout = 600000 |
如果只配置了 readTimeout 或 connectTimeout 的其中一條,配置不生效。
問題描述
Feign Client 配置如下:
1
2
3
|
feign.client.config. default .connectTimeout = 3000 feign.client.config. default .readTimeout = 3000 feign.client.config.my-api.readTimeout = 600000 |
默認超時時間配置了 3 秒。
由于 my-api 的響應數據較大,于是配置了 10 分鐘超時時間。
執行后發現,my-api 請求仍然 3 秒后就拋出如下異常:
1
2
3
4
5
6
7
8
|
Caused by: feign.RetryableException: Read timed out executing POST http: //my-api at feign.FeignException.errorExecuting(FeignException.java: 67 ) at feign.SynchronousMethodHandler.executeAndDecode(SynchronousMethodHandler.java: 104 ) at feign.SynchronousMethodHandler.invoke(SynchronousMethodHandler.java: 76 ) at feign.hystrix.HystrixInvocationHandler$ 1 .run(HystrixInvocationHandler.java: 108 ) at com.netflix.hystrix.HystrixCommand$ 2 .call(HystrixCommand.java: 302 ) at com.netflix.hystrix.HystrixCommand$ 2 .call(HystrixCommand.java: 298 ) at rx.internal.operators.OnSubscribeDefer.call(OnSubscribeDefer.java: 46 ) |
Feign Client的各種超時時間設置
在Spring Cloud微服務架構中,大部分公司都是利用Open Feign進行服務間的調用,而比較簡單的業務使用默認配置是不會有多大問題的,但是如果是業務比較復雜,服務要進行比較繁雜的業務計算,那后臺很有可能會出現Read Timeout這個異常,因此定制化配置超時時間就有必要了。
1. Feign Client Configuration
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
# 默認開啟 feign.httpclient.enabled= false # 默認關閉 feign.okhttp.enabled= true # 默認關閉 feign.hystrix.enabled= false # 默認關閉 feign.sentinel.enabled= true # default context 連接超時時間 feign.client.config. default .connectTimeout = 5000 # default context 讀超時時間 feign.client.config. default .readTimeout = 10000 # 設置重試處理器,默認直接拋出異常 # feign.client.config. default .retryer = Class<Retryer> # 設置日志級別,默認NONE # feign.client.config. default .loggerLevel = FULL |
2. Hystrix Configuration
1
2
|
# 全局設置超時: hystrix.command. default .execution.isolation.thread.timeoutInMilliseconds: 30000 |
hystrix在ribbon的外層處理。
3. Ribbon Configuration
1
2
3
4
5
6
|
# 連接超時時間,默認為 1 秒,該值會被FeignClient配置connectTimeout覆蓋 ribbon.ConnectTimeout= 5000 # 讀超時時間,默認為 1 秒,該值會被FeignClient配置readTimeout覆蓋 ribbon.ReadTimeout= 5000 # 最大重試次數 ribbon.MaxAutoRetries= 1 |
當Ribbon調用接口發送連接異常或者超時異常時會觸發Ribbon 的重試機制。
4. OkHttp Client Configuration
所設置的連接時間和超時時間最后會動態設置到OkHttpClient中,最底層也就是Socket的連接時間和讀超時時間。也就是說,直接配置OkHttpClient是無效的。
解決方案:添加OkHttp Client的請求Interceptor,動態設置超時時間。
1
2
3
4
5
6
7
8
9
10
11
12
|
@Bean ( "okHttpClient" ) public OkHttpClient okHttpClient(ConnectionPool connectionPool) { return new OkHttpClient().newBuilder().connectionPool(connectionPool) // 改值在FeignClient體系中會被動態覆蓋 .connectTimeout( 6 , TimeUnit.SECONDS) // 改值在FeignClient體系中會被動態覆蓋 .readTimeout(VmcConstants.TEN_SECONDS, TimeUnit.SECONDS) // 添加攔截器,支持動態設置超時時間 .addInterceptor( new OkHttpClientDynamicTimeoutInterceptor()) .eventListener(eventListener()) .build(); } |
5. 小結一下吧
1.如何配置好Hystrix和Ribbon的超時時間呢?
其實是有套路的,因為Feign的請求:其實是Hystrix+Ribbon。Hystrix在最外層,然后再到Ribbon,最后里面的是http請求。所以說。Hystrix的熔斷時間必須大于Ribbon的 ( ConnectTimeout + ReadTimeout )。而如果Ribbon開啟了重試機制,還需要乘以對應的重試次數,保證在Ribbon里的請求還沒結束時,Hystrix的熔斷時間不會超時。
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持服務器之家。
原文鏈接:https://blog.csdn.net/wu_weijie/article/details/112731675