content:[2021-02-26 18:43:59.939][http-nio-8080-exec-11][ERROR][c.l.c.c.r.RestCartServiceImpl:50] [reqId:7C0AB48A63C843C99CE5BC6A4ED85244] <getMyCartDetail> 操作异常: feign.RetryableException: Read timed out executing POST http://cic-product-service/cic-product/getProductListByIds at feign.FeignException.errorExecuting(FeignException.java:249) at feign.SynchronousMethodHandler.executeAndDecode(SynchronousMethodHandler.java:129) at feign.SynchronousMethodHandler.invoke(SynchronousMethodHandler.java:89) at feign.ReflectiveFeignProxy162.getProductListByIds(Unknown Source)
问题:在application.yaml中配置的feign的超时一直不生效,在网上也找不到相应的合理解决方案,通常的答案是对feign依赖的底层ribbon设置超时来解决,但这个不是官方推荐的方式,所以我就产生了跟踪源码的兴趣。
解决方案:首先我是通过对@FeignClient注解源码进行了断点跟踪,看下在服务启动时,spring boot是如何加载OpenFeign的超时配置的,最后跟踪到
image.png
FeignClientFactoryBean的configureUsingProperties方法读取了feign的配置
feign:
okhttp:
enabled: true
httpclient:
#关闭默认的httpclient
enabled: false
#整个连接池最大连接数
max-connections: 200
#每个路由的最大连接,默认50
max-connections-per-route: 50
client:
config:
default: #使用默认的feignclient时,超时可以生效
connectTimeout: 111000
readTimeout: 111000
image.png
跟踪到这儿,就可以知道在我们对@FeignClient不进行重写或者不额外指定configuration的配置时,他是默认加载default配置,而这个default配置默认超时是2秒
image.png
要想更改这个默认超时时间需要按照官方推荐的格式,才能生效,如下:
feign:
client:
config:
default: #这里就是指的所有被加载的默认FeignClient实现的服务配置都生效
connectTimeout: 111000
readTimeout: 111000
按照配置设置后,又对该服务配置做了验证和压测,在4核8g的mac本上,查询购物车服务支持150个并发,异常率0%,比之前的70%异常率,大大降低,至此完美解决线上问题。
网友评论