一、等待策略相关类:

1、等待策略接口:WaitStrategy接口

该接口只有一个方法,就是返回尝试失败之后,下一次尝试之前的等待时间。
long computeSleepTime(Attempt failedAttempt); 

2、创建等待策略对象的工厂类:com.github.rholder.retry.WaitStrategies类,这是一个常量类,负责创建等待策略对象。

  在该工厂类的内部实现了七种等待策略,如下:

  

二、七种等待策略详解如下:

1、等待固定时间的策略:FixedWaitStrategy类 (默认策略,等待时间为0L)

  这个类很简单,方法抛出异常后,隔固定时间进行一次重试即可。

  // 方法抛出异常后,conputeSleepTime返回固定的时间,单位ms
  @Override
  public long computeSleepTime(Attempt failedAttempt) {
  return sleepTime;
  }

2、等待时间自增的策略:IncrementingWaitStrategy类

  这个类每次重试的间隔时间随着重试次数的增加而增加,每次增加固定的时间。

   // initialSleepTime 默认时间
// increment 自增时间
// getAttemptNumber 第几次重试
    @Override
public long computeSleepTime(Attempt failedAttempt) {
long result = initialSleepTime + (increment * (failedAttempt.getAttemptNumber() - 1));
return result >= 0L ? result : 0L;
}

3、等待时间随机的策略:RandomWaitStrategy类

  这种策略等待时间在某个范围:

    如果传入一个参数,则为 [0L,timeUnit.toMillis(maximumTime)];

    如果传入两个个参数,则为 [minimumTimeUnit.toMillis(minimumTime),maximumTimeUnit.toMillis(maximumTime)]

  

4、等待时间指数形式增长:ExponentialWaitStrategy类

  这中策略等待时间呈指数形式增长,指数形式增长,如果指定最大等待时间,则增长到最大等待时间就不再增长;如果没有指定最大等待时间,则最大等待时间为Long.MAX_VALUE

  主要逻辑如下:

  

5、等待时间以斐波拉契数列形式增长:FibonacciWaitStrategy类

  有三个工厂方法:

    无参:等待时间从 1 增长到 Long.MAX_VALUE

    两个参数:等待时间从 1 增长到 maximumTimeUnit.toMillis(maximumTime),到最大值以后等待时间恒定不变

    三个参数:等待时间从 multiplier 增长到 maximumTimeUnit.toMillis(maximumTime)

  等待时间增长主要实现逻辑:

 

6、如果抛出的是指定异常,则从传入的函数中取得等待时间:ExceptionWaitStrategy类

  如果抛出的是指定异常,则从传入的function中取得等待时间并返回。如果异常不匹配,则返回等待时间为0L。

  具体实现逻辑如下:

    其中lastAttempt.getExceptionCause() 为抛出的异常;exceptionClass为指定异常。

   

  这里简单解释一下 Class.isAssignableFrom(Class clazz) 方法:该方法用来判断一个类Class1和另一个类Class2是否相同或者 Class1是Class2的父类或接口。    

 interface DemoInterface { }

 class DemoParent { }

 public class Demo extends DemoParent implements DemoInterface{

     public static void main(String[] args) {

         System.out.println(Demo.class.isAssignableFrom(DemoParent.class));    // false
System.out.println(Demo.class.isAssignableFrom(DemoInterface.class)); // false System.out.println(DemoParent.class.isAssignableFrom(Demo.class)); // true
System.out.println(DemoInterface.class.isAssignableFrom(Demo.class)); // true System.out.println(Demo.class.isAssignableFrom(Demo.class)); // true
}
}

7、组合多种等待策略,得到等待时间的方式:CompositeWaitStrategy类

  在获取等待时间时会获取多种等待策略各自的等待时间,然后累加这些等待时间并返回。

 /**
* 组合等待策略
* @author yjp
*/
@Test
public void testCompositeWaitStrategy() {
WaitStrategy fibonacciWait = WaitStrategies.fibonacciWait(1000L, 50000L, TimeUnit.MILLISECONDS);
WaitStrategy fixedWait = WaitStrategies.fixedWait(1000L, TimeUnit.MILLISECONDS);
WaitStrategy compositeWait = WaitStrategies.join(fibonacciWait, fixedWait);
assertEquals(2000L, compositeWait.computeSleepTime(failedAttempt(1, 1)));
// 第5次斐波拉契增长的等待时间是 5000L, 加上固定的等待时间 1000L
assertEquals(6000L, compositeWait.computeSleepTime(failedAttempt(5, 1)));
}
  
// --------------源码具体实现----------------
@Override
public long computeSleepTime(Attempt failedAttempt) {
long waitTime = 0L;
for (WaitStrategy waitStrategy : waitStrategies) {
     // 各策略根据各自的逻辑获取自己的等待是时间。然后累加到result中
waitTime += waitStrategy.computeSleepTime(failedAttempt);
}
return waitTime;
}
												

guava-retrying 源码解析(等待策略详解)的更多相关文章

  1. 【源码解析】BlockManager详解

    1 Block管理模块的组件和功能 BlockManager:BlockManager源码解析 Driver和Executor都会创建 Block的put.get和remove等操作的实际执行者 Bl ...

  2. vuex 源码解析(四) mutation 详解

    mutation是更改Vuex的store中的状态的唯一方法,mutation类似于事件注册,每个mutation都可以带两个参数,如下: state ;当前命名空间对应的state payload ...

  3. 我的书籍《深入解析Java编译器:源码剖析与实例详解》就要出版了

    一个十足的技术迷,2013年毕业,做过ERP.游戏.计算广告,在大公司呆过,但终究不满足仅对技术的应用,在2018年末离开了公司,全职写了一本书<深入解析Java编译器:源码剖析与实例详解> ...

  4. nginx源码分析线程池详解

    nginx源码分析线程池详解 一.前言     nginx是采用多进程模型,master和worker之间主要通过pipe管道的方式进行通信,多进程的优势就在于各个进程互不影响.但是经常会有人问道,n ...

  5. 【集合框架】JDK1.8源码分析之ArrayList详解(一)

    [集合框架]JDK1.8源码分析之ArrayList详解(一) 一. 从ArrayList字表面推测 ArrayList类的命名是由Array和List单词组合而成,Array的中文意思是数组,Lis ...

  6. Spring源码之九finishRefresh详解

    Spring源码之九finishRefresh详解 公众号搜索[程序员田同学],专职程序员兼业余写手,生活不止于写代码 Spring IoC 的核心内容要收尾了,本文将对最后一个方法 finishRe ...

  7. 第二章 Google guava cache源码解析1--构建缓存器

    1.guava cache 当下最常用最简单的本地缓存 线程安全的本地缓存 类似于ConcurrentHashMap(或者说成就是一个ConcurrentHashMap,只是在其上多添加了一些功能) ...

  8. Google guava cache源码解析1--构建缓存器(1)

    此文已由作者赵计刚授权网易云社区发布. 欢迎访问网易云社区,了解更多网易技术产品运营经验. 1.guava cache 当下最常用最简单的本地缓存 线程安全的本地缓存 类似于ConcurrentHas ...

  9. Guava Cache源码解析

    概述: 本次主要是分析cache的源码,基本概念官方简介即可. 基本类图: 在官方的文档说明中,Guava Cache实现了三种加载缓存的方式: LoadingCache在构建缓存的时候,使用buil ...

随机推荐

  1. Kubernetes容器上下文环境

    目录贴:Kubernetes学习系列 下面我们将主要介绍运行在Kubernetes集群中的容器所能够感知到的上下文环境,以及容器是如何获知这些信息的. 首先,Kubernetes提供了一个能够让容器感 ...

  2. js中prototype,constructor的理解

    连看4篇前辈的文章,记录一些知识点 Javascript继承机制的设计思想 Javascript 面向对象编程(一):封装 Javascript面向对象编程(二):构造函数的继承 Javascript ...

  3. 处理table 超出部分滚动问题

    我们有个需求是这样的,鉴于我的表达能力还是直接上原型图吧 今天主要记录上面的第四条解决过程. 首先我们的布局使用的table,当想给tbody设置高度的时候,发现不起作用.原因是table的默认是di ...

  4. C++读写图片数据转成Base64格式的一种方法

    最近在一个项目中要实现在客户端和服务端之间传送图片文件的功能,采用了C++语言读写图片转化成Base64格式进行传输.具体代码如下: //++Base64.h #pragma once class C ...

  5. Lombok 在继承类上面的注意

    找不到符号异常 在控制台项目中配置Lombok 编译会报错 找不到符号 解决 方法: 父类字段ToString不了 @Data @NoArgsConstructor @AllArgsConstruct ...

  6. Learning-Python【10】:函数初识

    一.什么是函数 函数就是代码的一种组织形式,是指将一组语句的集合通过一个名字(函数名)封装起来,要想指向这个函数,只需要调用其函数名即可 函数分为两大类:内置函数 和 自定义函数 二.为何要用函数 减 ...

  7. 函数func_get_args详解

    func_get_args ------获取一个函数的所有参数 function foo() { $numargs = func_num_args(); //参数数量 echo "参数个数是 ...

  8. Java 设置PDF文档背景——单色背景、图片背景

    一般生成的PDF文档默认的文档底色为白色,我们可以通过一定方法来更改文档的背景色,以达到文档美化的作用. 以下内容提供了Java编程来设置PDF背景色的方法.包括2种设置方法: 设置纯色背景色 设置图 ...

  9. Fastjson-fastjson中$ref对象重复引用问题

    当你有城市数据,你需要按国内.国际.热门城市分成数组的形式给出并输出为json格式. 第一个问题,你的数据格式,需要按字母类别划分,比如: "int": { "C&quo ...

  10. 【异常】idea执行Main方法出现 Exception in thread "main" java.lang.NoClassDefFoundError: javax/servlet/http/HttpServletRequest

    一.异常复现步骤 1)首先得是一个Spring MVC项目 注:Spring Boot项目有内置的web 容器,不会出现该问题 2)main方法存在于使用HttpServletRequest类的类中 ...