OpenFeign重试组件Retryer原理

简介: 该文章主要讲解了OpenFeign中的重试组件Retryer的工作原理及其实现细节。

前言

前面三篇文章已经分析了Feign的通信Client组件,Client是Rpc框架中最核心关键的能力,通过Client发起远程通信。

# OpenFeign第一个可扩展组件通信Client详解
# OpenFeign最核心组件LoadBalancerFeignClient详解
# OpenFeign集成Ribbon负载均衡-过滤和选择服务核心实现

市面上的Rpc框架在核心通信能力的基础上,有很多服务治理的组件,OpenFeign也一样,今天主角是重试组件,我们看看OpenFeign是如何实现重试的。

重试组件入口

OpenFeign执行远程请求是在feign.SynchronousMethodHandler#invoke,重试组件也是在这里嵌入的,我们看看具体实现

//调用远程入口
public Object invoke(Object[] argv) throws Throwable {
   
   
        RequestTemplate template = this.buildTemplateFromArgs.create(argv);
        Request.Options options = this.findOptions(argv);
        Retryer retryer = this.retryer.clone();
        //死循环,如果成功或者重试结束就返回
        while(true) {
   
   
            try {
   
   
                //通过client发起通信
                return this.executeAndDecode(template, options);
            } catch (RetryableException var9) {
   
   
                //处理RetryableException异常
                RetryableException e = var9;

                try {
   
   
                    //重试组件生效,判断是否重试
                    retryer.continueOrPropagate(e);
                } catch (RetryableException var8) {
   
   
                    Throwable cause = var8.getCause();
                    if (this.propagationPolicy == ExceptionPropagationPolicy.UNWRAP && cause != null) {
   
   
                        throw cause;
                    }

                    throw var8;
                }
            }
        }
}

发起请求的地方捕获了RetryableException异常,并且执行了重试的逻辑。 Feign提供了两个重试实现

1、feign.Retryer.Default

2、feign.Retryer#NEVER_RETRY

image.png

OpenFeign默认使用不重试

从openfeign的装配类FeignClientsConfiguration我们可以看出 在OpenFeign中,默认使用的是NEVER_RETRY,不重试

@Configuration
public class FeignClientsConfiguration {
   
   
    @Bean
    @ConditionalOnMissingBean
    public Retryer feignRetryer() {
   
   
        return Retryer.NEVER_RETRY;
    }
}

OpenFeign默认重试组件分析

feign.Retryer.Default是OpenFeign实现了重试能力的重试组件,我们看下他是如何实现的

public static class Default implements Retryer {
   
   
        private final int maxAttempts;
        private final long period;
        private final long maxPeriod;
        int attempt;
        long sleptForMillis;

        public Default() {
   
   
            this(100L, TimeUnit.SECONDS.toMillis(1L), 5);
        }

        public Default(long period, long maxPeriod, int maxAttempts) {
   
   
            this.period = period;
            this.maxPeriod = maxPeriod;
            this.maxAttempts = maxAttempts;
            this.attempt = 1;
        }

        protected long currentTimeMillis() {
   
   
            return System.currentTimeMillis();
        }

        public void continueOrPropagate(RetryableException e) {
   
   
            //已经超过最大重试次数,不重试了,异常往上抛
            //attempt默认是1
            if (this.attempt++ >= this.maxAttempts) {
   
   
                throw e;
            } else {
   
   
                long interval;
                //根据上次重试时间计算间隔时间
                if (e.retryAfter() != null) {
   
   
                    interval = e.retryAfter().getTime() - this.currentTimeMillis();
                    //间隔时间是否超过最大重试间隔时间
                    if (interval > this.maxPeriod) {
   
   
                        interval = this.maxPeriod;
                    }

                    if (interval < 0L) {
   
   
                        return;
                    }
                } else {
   
   
                    //获取下次重试间隔时间
                    interval = this.nextMaxInterval();
                }

                try {
   
   
                    //睡眠重试间隔,再发起重试
                    Thread.sleep(interval);
                } catch (InterruptedException var5) {
   
   
                    Thread.currentThread().interrupt();
                    throw e;
                }

                this.sleptForMillis += interval;
            }
        }

        long nextMaxInterval() {
   
   
            //计算下次重试间隔,按照1.5为底的指数递增
            long interval = (long)((double)this.period * Math.pow(1.5, (double)(this.attempt - 1)));
            return interval > this.maxPeriod ? this.maxPeriod : interval;
        }

        public Retryer clone() {
   
   
            return new Default(this.period, this.maxPeriod, this.maxAttempts);
        }
    }

1、默认重试是捕获RetryableException异常,如果是RetryableException异常则执行重试逻辑。

2、每次重试间隔按照底数是1.5的指数级递增

自定义重试组件

我们可以自己实现重试组件feign.Retryer接口,也可以使用OpenFeign自带的重试组件feign.Retryer.Default,假如我们使用自带的重试组件,我们只要定义一个配置类,配置Retryer的Bean,覆盖默认的Retryer.NEVER_RETRY;

下方是配置每次重试间隔是100ms,最大重试间隔是1s, 重试次数是2次

@Component
public class CustomFeignConfig {
   
   


    public Integer retryPeriod = 100;
    public Integer retryMaxPeriod = 1000;
    public Integer maxRetryAttempts = 2;

    //自定义重试组件,覆盖OpenFeign默认的Retryer.NEVER_RETRY;
    @Bean
    public Retryer retryer() {
   
   
        return new Retryer.Default(retryPeriod, retryMaxPeriod, maxRetryAttempts);
    }
}

通过上面的配置类,OpenFeign就具备重试功能了,请注意虽然maxRetryAttempts配置是2,但是只会重试1次,2次包括本身调用那一次。也就是说,如果想除去本身调用那次外重试2次,则maxRetryAttempts需要配置为3。

总结

1、OpenFeign提供了重试组件,默认是不重试的

2、feign.Retryer.Default的实际重试次数是maxAttempts-1

OpenFeign重试组件分析完了,对你有没有帮助呢?

image.png

相关文章
|
NoSQL Java Redis
Seata常见问题之实现openfeign远程调用失败如何解决
Seata 是一个开源的分布式事务解决方案,旨在提供高效且简单的事务协调机制,以解决微服务架构下跨服务调用(分布式场景)的一致性问题。以下是Seata常见问题的一个合集
Seata常见问题之实现openfeign远程调用失败如何解决
|
Java 数据库 网络架构
Hystrix使用及其配置详解
Hystrix使用及其配置详解
1798 0
Hystrix使用及其配置详解
|
分布式计算 Java 数据库连接
了解Spring R2DBC的声明式事务实现机制
# Spring非反应式事务实现原理 Spring基于注解和AOP的声明式事务(@Transactional)已经是业务开发的常用工具,默认是采用同步的方式基于ThreadLocal(保存连接信息和会话信息等)实现,在具体数据库操作时就使用同一个数据库连接,并手动提交事务,保证数据正确性。 # 基于反应式的Spring事务有何不同 Spring的反应式实现是基于Reactor框架,该框架
2993 0
|
3月前
|
安全 Java 数据库
Jasypt加密数据库配置信息
本文介绍了使用 Jasypt 对配置文件中的公网数据库认证信息进行加密的方法,以提升系统安全性。主要内容包括:1. 背景介绍;2. 前期准备,如依赖导入及版本选择;3. 生成密钥并实现加解密测试;4. 在配置文件中应用加密后的密码,并通过测试接口验证解密结果。确保密码安全的同时,保障系统的正常运行。
286 3
Jasypt加密数据库配置信息
|
5月前
|
负载均衡 前端开发 Java
SpringCloud调用组件Feign
本文深入探讨微服务Spring体系中的Feign组件。Feign是一个声明式Web服务客户端,支持注解、编码器/解码器,与Spring MVC注解兼容,并集成Eureka、负载均衡等功能。文章详细介绍了SpringCloud整合Feign的步骤,包括依赖引入、客户端启用、接口创建及调用示例。同时,还涵盖了Feign的核心配置,如超时设置、拦截器实现(Basic认证与自定义)和日志级别调整。最后,总结了`@FeignClient`常用属性,帮助开发者更好地理解和使用Feign进行微服务间通信。
483 1
|
缓存 负载均衡 Java
OpenFeign第一个可扩展组件通信Client详解
这篇文章详细分析了OpenFeign框架中的第一个可扩展组件——通信Client,包括其默认实现`feign.Client.Default`,以及如何使用`LoadBalancerFeignClient`集成负载均衡能力,并探讨了如何替换默认的`HttpURLConnection`通信组件为`OkHttpClient`或`ApacheHttpClient`。
OpenFeign第一个可扩展组件通信Client详解
|
消息中间件 JSON Java
Spring Boot、Spring Cloud与Spring Cloud Alibaba版本对应关系
Spring Boot、Spring Cloud与Spring Cloud Alibaba版本对应关系
23918 0
|
存储 监控 Java
OpenFeign请求拦截器组件RequestInterceptor原理与使用场景
该文章讲述了OpenFeign中的请求拦截器组件RequestInterceptor的原理及其常见使用场景。
OpenFeign请求拦截器组件RequestInterceptor原理与使用场景
|
缓存 负载均衡 Java
OpenFeign最核心组件LoadBalancerFeignClient详解(集成Ribbon负载均衡能力)
文章标题为“OpenFeign的Ribbon负载均衡详解”,是继OpenFeign十大可扩展组件讨论之后,深入探讨了Ribbon如何为OpenFeign提供负载均衡能力的详解。
OpenFeign最核心组件LoadBalancerFeignClient详解(集成Ribbon负载均衡能力)
|
消息中间件 存储 canal
3分钟白话RocketMQ系列—— 如何发送消息
3分钟白话RocketMQ系列—— 如何发送消息
524 0