在Spring Cloud
微服务架构中,大部分公司都是利用Open Feign
进行服务间的调用,而比较简单的业务使用默认配置是不会有多大问题的,但是如果是业务比较复杂,服务要进行比较繁杂的业务计算,那后台很有可能会出现Read Timeout
这个异常,因此定制化配置超时时间就有必要了。
Feign Client Configuration
# 默认开启
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
Hystrix Configuration
# 全局设置超时:
hystrix.command.default.execution.isolation.thread.timeoutInMilliseconds: 30000
hystrix在ribbon的外层处理。
Ribbon Configuration
# 连接超时时间,默认为1秒,该值会被FeignClient配置connectTimeout覆盖
ribbon.ConnectTimeout=5000
# 读超时时间,默认为1秒,该值会被FeignClient配置readTimeout覆盖
ribbon.ReadTimeout=5000
# 最大重试次数
ribbon.MaxAutoRetries=1
当Ribbon调用接口发送连接异常或者超时异常时会触发Ribbon 的重试机制。
OkHttp Client Configuration
所设置的连接时间和超时时间最后会动态设置到OkHttpClient中,最底层也就是Socket的连接时间和读超时时间。也就是说,直接配置OkHttpClient是无效的。
解决方案:添加OkHttp Client的请求Interceptor
,动态设置超时时间。
@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();
}
总结
1.如何配置好Hystrix
和Ribbon
的超时时间呢?
其实是有套路的,因为Feign
的请求:其实是Hystrix
+Ribbon
。Hystrix
在最外层,然后再到Ribbon
,最后里面的是http
请求。所以说。Hystrix
的熔断时间必须大于Ribbon
的 ( ConnectTimeout
+ ReadTimeout
)。而如果Ribbon
开启了重试机制,还需要乘以对应的重试次数,保证在Ribbon
里的请求还没结束时,Hystrix
的熔断时间不会超时。
参考资料
【2】Feign的各种超时时间