springcloud之RestTemplate、Ribbon、Feign之间的超时和重试及关系

客官°小女子只卖身不卖艺 2022-01-22 06:43 832阅读 1赞

RestTemplate和Riboon超时关系

在SpringCloud项目中,RestTemplate一般通过负责均匀来使用,其超时时间不能通过ribbon直接来设置,这样设置是无效的。如:

  1. #ribbon的超时时间
  2. ribbon:
  3. ReadTimeout: 3000 #读取超时时间
  4. ConnectTimeout: 3000 #连接超时时间

restTemplate的超时时间可以在具体的客户端指定(java原生、apache客户端、netty、okhttp),这里用Java原生的实现来举例。

  1. @Bean
  2. public RestTemplate getRestTemplate(){
  3. //default
  4. SimpleClientHttpRequestFactory requestFactory =
  5. new SimpleClientHttpRequestFactory();
  6. requestFactory.setReadTimeout(15000);//读取超时
  7. requestFactory.setConnectTimeout(5000);//连接超时
  8. RestTemplate restTemplate = new RestTemplate(requestFactory);
  9. return restTemplate;
  10. }

更多的客户端实现参考
HTTP协议参考

Feign超时时间 和重试

使用Feign调用接口分两层,ribbon的调用和hystrix的调用,所以ribbon的超时时间和Hystrix的超时时间的结合就是Feign的超时时间。

  1. #hystrix 生效标志,不同的springcloud版本可能配置不同
  2. feign:
  3. hystrix:
  4. enabled: true
  5. #hystrix的超时时间
  6. hystrix:
  7. command:
  8. default:
  9. execution:
  10. timeout:
  11. enabled: true
  12. isolation:
  13. thread:
  14. timeoutInMilliseconds: 9000
  15. #ribbon的超时时间
  16. ribbon:
  17. ReadTimeout: 3000
  18. ConnectTimeout: 3000

涉及到ribbon的重试机制,一般情况下,ribbon 的超时时间(<)hystrix的超时时间。
因为ribbon的重试机制和Feign的重试机制有冲突,所以源码中默认关闭Feign的重试机制,源码如下(这里以springcloud版本:Edgware.SR5为例):
FeignClientsConfiguration类相关的配置代码

  1. @Configuration
  2. public class FeignClientsConfiguration {
  3. ... 省略部分代码
  4. /** 这里是初始化重试的实现类,Retryer.NEVER_RETRY 默认feign不进行重试机制 */
  5. @Bean
  6. @ConditionalOnMissingBean
  7. public Retryer feignRetryer() {
  8. return Retryer.NEVER_RETRY;
  9. }
  10. ... 代码省略
  11. }
  12. public interface Retryer extends Cloneable {
  13. //Retryer 的不进行重试的实现类
  14. Retryer NEVER_RETRY = new Retryer() {
  15. public void continueOrPropagate(RetryableException e) {
  16. throw e;
  17. }
  18. public Retryer clone() {
  19. return this;
  20. }
  21. };
  22. void continueOrPropagate(RetryableException var1);
  23. Retryer clone();
  24. // Retryer 重试的实现类,默认重试5次
  25. public static class Default implements Retryer {
  26. private final int maxAttempts;
  27. private final long period;
  28. private final long maxPeriod;
  29. int attempt;
  30. long sleptForMillis;
  31. public Default() {
  32. this(100L, TimeUnit.SECONDS.toMillis(1L), 5);
  33. }
  34. public Default(long period, long maxPeriod, int maxAttempts) {
  35. this.period = period;
  36. this.maxPeriod = maxPeriod;
  37. this.maxAttempts = maxAttempts;
  38. this.attempt = 1;
  39. }
  40. }
  41. }

feign默认关闭了重试,使用了 Retryer NEVER_RETRY = new Retryer()的实现。
如若开启feign重试:

@Bean
Retryer feignRetryer() {
return new Retryer.Default();
}

ribbon重试机制

  1. ribbon:
  2. # http建立socket超时时间,毫秒
  3. ConnectTimeout: 2000
  4. # http读取响应socket超时时间
  5. ReadTimeout: 10000
  6. # 同一台实例最大重试次数,不包括首次调用
  7. MaxAutoRetries: 1
  8. # 重试负载均衡其他的实例最大重试次数,不包括首次server
  9. MaxAutoRetriesNextServer: 1
  10. # 是否所有操作都重试,POST请求注意多次提交错误。
  11. # 默认false,设定为false的话,只有get请求会重试
  12. OkToRetryOnAllOperations: true

在feign开启的情况下:
ribbon重试计算公式:MaxAutoRetries+MaxAutoRetriesNextServer+(MaxAutoRetries *MaxAutoRetriesNextServer) ,即重试3次 (不包括首次调用)一共产生4次调用。
如果在重试期间,时间超过了hystrix的超时时间,便会立即执行熔断,fallback。根据上面配置的参数计算hystrix的超时时间>=ReadTimeout * 重试次数,不然重试机制就会没有意义 。
hystrix超时时间的计算: (1 + MaxAutoRetries + MaxAutoRetriesNextServer) * ReadTimeout 即按照以上的配置 hystrix的超时时间应该配置为 (1+1+1)*10=30秒。
当ribbon超时后且hystrix没有超时,便会采取重试机制。当OkToRetryOnAllOperations设置为false时,只会对get请求进行重试。如果设置为true,便会对所有的请求进行重试,如果是put或post等写操作,如果服务器接口没做幂等性,会产生不好的结果,所以OkToRetryOnAllOperations慎用。
如果不配置ribbon的重试次数,默认会重试一次

注意:
1、默认情况下,GET方式请求无论是连接异常还是读取异常,都会进行重试
非GET方式请求,只有连接异常时,才会进行重试。
2、feign肯定会用到ribbon,建议在配置中关闭feign的Hystrix重试和ribbon重试机制,避免产生不良的影响。

参考:ribbon
ribbon 重试源码
feign 解析

发表评论

表情:
评论列表 (有 0 条评论,832人围观)

还没有评论,来说两句吧...

相关阅读

    相关 超时机制

    网上发现这篇好文章,这里记录学习。 介绍       在实际开发过程中,笔者见过太多故障是因为超时没有设置或者设置的不对而造成的。而这些故障都是因为没有意识到超时设置的