Java函数调用重试的正确姿势

简介: Java函数调用重试的正确姿势

1、引言

业务开发中很可能与回到重试的场景。


重试主要在调用失败时重试,尤其是发生dubbo相关异常,网络相关异常的时候。


下面对该功能简单作封装,然后给出一些相对用的多一些的开源代码地址。



核心功能

提供重试工具类,

支持传入操作、重试次数和延时时间。

支持定义不再重试的异常和条件。


主要应用场景

只要适用于对任务丢失要求不高的场景。

此工具类只适合单机版,因此任务的丢失要求高的场景建议用中间件,如缓存中间件redis或者消息中间件。


主要场景如下:

- 乐观锁重试

- 上游业务保证重试的场景且没有其他好的重试机制

- 需要轮询直到得到想要的结果的场景

- 其他需要控制重试时间间隔的场景



2、简单封装

github地址 https://github.com/chujianyun/simple-retry4j


maven依赖


https://search.maven.org/search?q=a:simple-retry4j



可下载运行,可fork改进,欢迎提出宝贵意见,欢迎贡献代码。



封装重试策略


package com.github.chujianyun.simpleretry4j;

import lombok.Data;

import org.apache.commons.collections4.CollectionUtils;

import java.time.Duration;

import java.util.ArrayList;

import java.util.List;

import java.util.concurrent.TimeUnit;

import java.util.function.Predicate;

/**

* 重试策略

*

* @author: 明明如月 liuwangyangedu@163.com

* @date: 2019-04-05 10:06

*/

@Data

public class RetryPolicy {

   /**

    * 最大重试次数(如果不设置则默认不满足重试的异常或策略则无限重试)

    */

   private Integer maxRetries;

   /**

    * 延时时间

    */

   private Duration delayDuration;

   /**

    * 不需要重试的异常列表

    */

   private List<Class<? extends Exception>> abortExceptions;

   /**

    * 不需要重试的条件列表(满足其中一个则不重试,如果要传入泛型条件是返回值或者其父类类型)

    */

   private List<Predicate> abortConditions;

   public RetryPolicy(Builder builder) {

       this.maxRetries = builder.maxRetries;

       this.delayDuration = builder.delayDuration;

       List<Class<? extends Exception>> abortExceptions = builder.abortExceptions;

       if (CollectionUtils.isEmpty(abortExceptions)) {

           this.abortExceptions = new ArrayList<>();

       } else {

           this.abortExceptions = abortExceptions;

       }

       List<Predicate> abortConditions = builder.abortConditions;

       if (CollectionUtils.isEmpty(abortConditions)) {

           this.abortConditions = new ArrayList<>();

       } else {

           this.abortConditions = abortConditions;

       }

   }

   public static Builder builder() {

       return new Builder();

   }

   public static class Builder {

       private Integer maxRetries;

       private Duration delayDuration;

       private List<Class<? extends Exception>> abortExceptions = new ArrayList<>();

       private List<Predicate> abortConditions = new ArrayList<>();

       /**

        * 设置最大重试次数(如果不设置则默认不满足重试的异常或策略则无限重试)

        */

       public Builder maxRetries(Integer maxRetries) {

           if (maxRetries == null || maxRetries < 0) {

               throw new IllegalArgumentException("maxRetries must not be null or negative");

           }

           this.maxRetries = maxRetries;

           return this;

       }

       /**

        * 重试的时间间隔

        */

       public Builder delayDuration(Duration delayDuration) {

           if (delayDuration == null || delayDuration.isNegative()) {

               throw new IllegalArgumentException("delayDuration must not be null or negative");

           }

           this.delayDuration = delayDuration;

           return this;

       }

       /**

        * 重试的时间间隔

        */

       public Builder delayDuration(Integer time, TimeUnit timeUnit) {

           if (time == null || time < 0) {

               throw new IllegalArgumentException("time must not be null or negative");

           }

           if (timeUnit == null) {

               throw new IllegalArgumentException("timeUnit must not be null or negative");

           }

           this.delayDuration = Duration.ofMillis(timeUnit.toMillis(time));

           return this;

       }

       /**

        * 设置不重试的策略列表

        */

       public Builder abortConditions(List<Predicate> predicates) {

           if (CollectionUtils.isNotEmpty(predicates)) {

               predicates.forEach(this::abortCondition);

           }

           return this;

       }

       /**

        * 新增不重试的策略

        */

       public Builder abortCondition(Predicate predicate) {

           if (predicate != null) {

               this.abortConditions.add(predicate);

           }

           return this;

       }

       /**

        * 设置不重试的异常列表

        */

       public Builder abortExceptions(List<Class<? extends Exception>> abortExceptions) {

           if (CollectionUtils.isNotEmpty(abortExceptions)) {

               abortExceptions.forEach(this::abortException);

           }

           return this;

       }

       /**

        * 新增不重试的异常

        */

       public Builder abortException(Class<? extends Exception> exception) {

           if (exception != null) {

               this.abortExceptions.add(exception);

           }

           return this;

       }

       public RetryPolicy build() {

           return new RetryPolicy(this);

       }

   }

}


封装重试工具类


package com.github.chujianyun.simpleretry4j;

import lombok.extern.slf4j.Slf4j;

import org.apache.commons.collections4.CollectionUtils;

import java.time.Duration;

import java.util.List;

import java.util.concurrent.Callable;

import java.util.function.Consumer;

import java.util.function.Predicate;

/**

* 方法重试工具类

*

* @author: 明明如月 liuwangyangedu@163.com

* @date: 2019-04-05 02:09

*/

@Slf4j

public class SimpleRetryUtil {

   /**

    * 无返回值的重试方法

    */

   public static <T> void executeWithRetry(Consumer<T> consumer, T data, RetryPolicy retryPolicy) throws Exception {

       executeWithRetry(null, consumer, data, retryPolicy);

   }

   /**

    * 带返回值的重试方法

    */

   public static <T> T executeWithRetry(Callable<T> callable, RetryPolicy retryPolicy) throws Exception {

       return executeWithRetry(callable, null, null, retryPolicy);

   }

   /**

    * 带重试和延时的操作执行

    *

    * @param callable    执行的操作

    * @param retryPolicy 重试策略

    * @return 返回值

    * @throws Exception 业务异常或者超过最大重试次数后的最后一次尝试抛出的异常

    */

   private static <T> T executeWithRetry(Callable<T> callable, Consumer<T> consumer, T data, RetryPolicy retryPolicy) throws Exception {

       // 最大重试次数

       Integer maxRetries = retryPolicy.getMaxRetries();

       if (maxRetries != null && maxRetries < 0) {

           throw new IllegalArgumentException("最大重试次数不能为负数");

       }

       int retryCount = 0;

       Duration delayDuration = retryPolicy.getDelayDuration();

       while (true) {

           try {

               // 不带返回值的

               if (consumer != null) {

                   consumer.accept(data);

                   return null;

               }

               //  带返回值的

               if (callable != null) {

                   T result = callable.call();

                   // 不设置终止条件或者设置了且满足则返回,否则还会重试

                   List<Predicate> abortConditions = retryPolicy.getAbortConditions();

                   /* ---------------- 不需要重试的返回值 -------------- */

                   if (isInCondition(result, abortConditions)) {

                       return result;

                   }

                   /* ---------------- 需要重试的返回值 -------------- */

                   boolean hasNextRetry = hasNextRetryAfterOperation(++retryCount, maxRetries, delayDuration);

                   if (!hasNextRetry) {

                       return result;

                   }

               }

           } catch (Exception e) {

               /* ---------------- 不需要重试的异常 -------------- */

               List<Class<? extends Exception>> abortExceptions = retryPolicy.getAbortExceptions();

               if (isInExceptions(e, abortExceptions)) {

                   throw e;

               }

               /* ---------------- 需要重试的异常 -------------- */

               boolean hasNextRetry = hasNextRetryAfterOperation(++retryCount, maxRetries, delayDuration);

               if (!hasNextRetry) {

                   throw e;

               }

           }

       }

   }

   /**

    * 判断运行之后是否还有下一次重试

    */

   private static boolean hasNextRetryAfterOperation(int retryCount, Integer maxRetries, Duration delayDuration) throws InterruptedException {

       // 有限次重试

       if (maxRetries != null) {

           if (retryCount > maxRetries) {

               return false;

           }

       }

       // 延时

       if (delayDuration != null && !delayDuration.isNegative()) {

           log.debug("延时{}毫秒", delayDuration.toMillis());

           Thread.sleep(delayDuration.toMillis());

       }

       log.debug("第{}次重试", retryCount);

       return true;

   }

   /**

    * 是否在异常列表中

    */

   private static boolean isInExceptions(Exception e, List<Class<? extends Exception>> abortExceptions) {

       if (CollectionUtils.isEmpty(abortExceptions)) {

           return false;

       }

       for (Class<? extends Exception> clazz : abortExceptions) {

           if (clazz.isAssignableFrom(e.getClass())) {

               return true;

           }

       }

       return false;

   }

   /**

    * 是否符合不需要终止的条件

    */

   private static <T> boolean isInCondition(T result, List<Predicate> abortConditions) {

       if (CollectionUtils.isEmpty(abortConditions)) {

           return true;

       }

       for (Predicate predicate : abortConditions) {

           if (predicate.test(result)) {

               return true;

           }

       }

       return false;

   }

}


遇到业务异常就没必要重试了,直接扔出去。


当遇到非业务异常是,未超出最大重试次数时,不断重试,如果设置了延时则延时后重试。


测试类


package com.github.chujianyun.simpleretry4j;

import com.github.chujianyun.simpleretry4j.exception.BusinessException;

import lombok.extern.slf4j.Slf4j;

import org.junit.Assert;

import org.junit.Test;

import org.junit.jupiter.api.TestInstance;

import org.junit.runner.RunWith;

import org.mockito.Mock;

import org.mockito.Mockito;

import org.powermock.modules.junit4.PowerMockRunner;

import java.time.Duration;

import java.util.ArrayList;

import java.util.List;

import java.util.Objects;

import java.util.concurrent.Callable;

import java.util.concurrent.TimeUnit;

import java.util.function.Consumer;

import static org.mockito.ArgumentMatchers.any;

/**

* 重试测试

*

* @author: 明明如月 liuwangyangedu@163.com

* @date: 2019-04-04 10:42

*/

@Slf4j

@RunWith(PowerMockRunner.class)

@TestInstance(TestInstance.Lifecycle.PER_CLASS)

public class SimpleRetryUtilTest {

   @Mock

   private Callable<Integer> callable;

   @Mock

   private Consumer<List<Integer>> consumer;

   /**

    * 提供两种设置延时时间的方法

    */

   @Test

   public void delayDuration() {

       RetryPolicy retryPolicy1 = RetryPolicy.builder()

               .maxRetries(3)

               .delayDuration(Duration.ofMillis(5))

               .build();

       RetryPolicy retryPolicy2 = RetryPolicy.builder()

               .maxRetries(3)

               .delayDuration(5, TimeUnit.MILLISECONDS)

               .build();

       Assert.assertEquals(retryPolicy1.getDelayDuration(), retryPolicy2.getDelayDuration());

   }

   /**

    * 模拟异常重试

    */

   @Test(expected = Exception.class)

   public void executeWithRetry_Exception() throws Exception {

       RetryPolicy retryPolicy = RetryPolicy.builder()

               .maxRetries(3)

               .build();

       Mockito.doThrow(new Exception("test")).when(callable).call();

       SimpleRetryUtil.executeWithRetry(callable, retryPolicy);

   }

   /**

    * 模拟异常重试

    */

   @Test(expected = BusinessException.class)

   public void executeWithRetry_BusinessException() throws Exception {

       RetryPolicy retryPolicy = RetryPolicy.builder()

               .maxRetries(3)

               .delayDuration(Duration.ofMillis(100))

               .build();

       Mockito.doThrow(new BusinessException()).when(callable).call();

       SimpleRetryUtil.executeWithRetry(callable, retryPolicy);

   }

   /**

    * 模拟终止异常不重试

    */

   @Test(expected = IllegalArgumentException.class)

   public void executeWithAbortException() throws Exception {

       RetryPolicy retryPolicy = RetryPolicy.builder()

               .maxRetries(3)

               .delayDuration(Duration.ofMillis(100))

               .abortException(IllegalArgumentException.class)

               .abortException(BusinessException.class)

               .build();

           Mockito.doThrow(new IllegalArgumentException()).doReturn(1).when(callable).call();

           Integer result = SimpleRetryUtil.executeWithRetry(callable, retryPolicy);

           log.debug("最终返回值{}", result);

   }

   /**

    * 模拟不在终止异常触发重试

    */

   @Test

   public void executeWithAbortException2() throws Exception {

       RetryPolicy retryPolicy = RetryPolicy.builder()

               .maxRetries(3)

               .delayDuration(Duration.ofMillis(100))

               .abortException(BusinessException.class)

               .build();

       Mockito.doThrow(new NullPointerException()).doReturn(1).when(callable).call();

       Integer result = SimpleRetryUtil.executeWithRetry(callable, retryPolicy);

       log.debug("最终返回值{}", result);

   }

   /**

    * 满足条件的返回值不重试的设置

    */

   @Test

   public void executeWithAbortCondition() throws Exception {

       RetryPolicy retryPolicy = RetryPolicy.builder()

               .maxRetries(3)

               .delayDuration(Duration.ofMillis(100))

               .abortCondition(Objects::nonNull)

               .build();

       //前两次返回null 需要重试

       Mockito.doReturn(null).doReturn(null).doReturn(1).when(callable).call();

       Integer result = SimpleRetryUtil.executeWithRetry(callable, retryPolicy);

       log.debug("最终返回值{}", result);

   }

   /**

    * 测试无返回值的情况

    */

   @Test

   public void consumerTest() throws Exception {

       RetryPolicy retryPolicy = RetryPolicy.builder()

               .maxRetries(3)

               .delayDuration(Duration.ofMillis(100))

               .build();

       List<Integer> data = new ArrayList<>(4);

       data.add(1);

       data.add(2);

       data.add(3);

       data.add(4);

       Mockito.doThrow(new RuntimeException("测试")).doThrow(new RuntimeException("测试2")).doAnswer(invocationOnMock -> {

           Object param = invocationOnMock.getArgument(0);

           System.out.println("消费成功,列表个数" + ((List) param).size());

           return param;

       }).when(consumer).accept(any());

       SimpleRetryUtil.executeWithRetry(consumer, data, retryPolicy);

   }

}

日志配置


# 设置

log4j.rootLogger = debug,stdout

# 输出信息到控制抬

log4j.appender.stdout = org.apache.log4j.ConsoleAppender

log4j.appender.stdout.Target = System.out

log4j.appender.stdout.layout = org.apache.log4j.PatternLayout

log4j.appender.stdout.layout.ConversionPattern = [%-5p] %d{yyyy-MM-dd HH:mm:ss,SSS} method:%l%n%m%n

pom文件


<?xml version="1.0" encoding="UTF-8"?>

<project xmlns="http://maven.apache.org/POM/4.0.0"

        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

        xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">

   <modelVersion>4.0.0</modelVersion>

   <groupId>com.github.chujianyun</groupId>

   <artifactId>simple-retry4j</artifactId>

   <version>1.1.2</version>

   <packaging>jar</packaging>

   <name>simple-retry4j</name>

   <description>A Java method retry and batch execute open source lib.</description>

   <url>https://github.com/chujianyun/simple-retry4j/tree/master</url>

   <properties>

       <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>

       <junit-jupiter.version>5.3.1</junit-jupiter.version>

       <maven.compiler.target>1.8</maven.compiler.target>

       <maven.compiler.source>1.8</maven.compiler.source>

   </properties>

   <dependencies>

       <!-- https://mvnrepository.com/artifact/org.slf4j/slf4j-api -->

       <dependency>

           <groupId>org.slf4j</groupId>

           <artifactId>slf4j-api</artifactId>

           <version>1.7.26</version>

       </dependency>

       <dependency>

           <groupId>log4j</groupId>

           <artifactId>log4j</artifactId>

           <version>1.2.17</version>

       </dependency>

       <dependency>

           <groupId>org.slf4j</groupId>

           <artifactId>slf4j-log4j12</artifactId>

           <version>1.7.25</version>

       </dependency>

       <dependency>

           <groupId>org.projectlombok</groupId>

           <artifactId>lombok</artifactId>

           <version>1.18.2</version>

       </dependency>

       <dependency>

           <groupId>junit</groupId>

           <artifactId>junit</artifactId>

           <version>4.11</version>

           <scope>test</scope>

       </dependency>

       <!-- https://mvnrepository.com/artifact/org.apache.commons/commons-lang3 -->

       <dependency>

           <groupId>org.apache.commons</groupId>

           <artifactId>commons-lang3</artifactId>

           <version>3.8.1</version>

       </dependency>

       <!-- https://mvnrepository.com/artifact/org.apache.commons/commons-collections4 -->

       <dependency>

           <groupId>org.apache.commons</groupId>

           <artifactId>commons-collections4</artifactId>

           <version>4.3</version>

       </dependency>

       <dependency>

           <groupId>org.junit.jupiter</groupId>

           <artifactId>junit-jupiter-engine</artifactId>

           <version>${junit-jupiter.version}</version>

           <scope>test</scope>

       </dependency>

       <dependency>

           <groupId>org.powermock</groupId>

           <artifactId>powermock-module-junit4</artifactId>

           <version>2.0.0</version>

           <scope>test</scope>

       </dependency>

       <dependency>

           <groupId>org.powermock</groupId>

           <artifactId>powermock-api-mockito2</artifactId>

           <version>2.0.0</version>

           <scope>test</scope>

       </dependency>

   </dependencies>

   <licenses>

       <license>

           <name>The Apache Software License, Version 2.0</name>

           <url>http://www.apache.org/licenses/LICENSE-2.0.txt</url>

           <distribution>repo</distribution>

       </license>

   </licenses>

   <developers>

       <developer>

           <name>liuwangyang</name>

           <email>liuwangyangedu@163.com</email>

           <organization>https://github.com/chujianyun</organization>

           <timezone>+8</timezone>

       </developer>

   </developers>

   <scm>

       <connection>scm:git:git@github.com:chujianyun/simple-retry4j.git</connection>

       <developerConnection>scm:git:git@github.com:chujianyun/simple-retry4j.git</developerConnection>

       <url>https://github.com/chujianyun/simple-retry4j/tree/master</url>

   </scm>

   <distributionManagement>

       <snapshotRepository>

           <id>ossrh</id>

           <url>https://oss.sonatype.org/content/repositories/snapshots</url>

       </snapshotRepository>

       <repository>

           <id>ossrh</id>

           <url>https://oss.sonatype.org/service/local/staging/deploy/maven2/</url>

       </repository>

   </distributionManagement>

   <build>

       <plugins>

           <plugin>

               <groupId>org.apache.maven.plugins</groupId>

               <artifactId>maven-source-plugin</artifactId>

               <version>2.2.1</version>

               <executions>

                   <execution>

                       <id>attach-sources</id>

                       <goals>

                           <goal>jar-no-fork</goal>

                       </goals>

                   </execution>

               </executions>

           </plugin>

           <plugin>

               <groupId>org.apache.maven.plugins</groupId>

               <artifactId>maven-javadoc-plugin</artifactId>

               <version>2.9.1</version>

               <configuration>

                   <show>private</show>

                   <nohelp>true</nohelp>

                   <charset>UTF-8</charset>

                   <encoding>UTF-8</encoding>

                   <docencoding>UTF-8</docencoding>

                   <additionalparam>-Xdoclint:none</additionalparam>  <!-- TODO 临时解决不规范的javadoc生成报错,后面要规范化后把这行去掉 -->

               </configuration>

               <executions>

                   <execution>

                       <id>attach-javadocs</id>

                       <goals>

                           <goal>jar</goal>

                       </goals>

                   </execution>

               </executions>

           </plugin>

           <plugin>

               <groupId>org.apache.maven.plugins</groupId>

               <artifactId>maven-gpg-plugin</artifactId>

               <version>1.5</version>

               <executions>

                   <execution>

                       <id>sign-artifacts</id>

                       <phase>verify</phase>

                       <goals>

                           <goal>sign</goal>

                       </goals>

                   </execution>

               </executions>

           </plugin>

           <plugin>

               <groupId>org.sonatype.plugins</groupId>

               <artifactId>nexus-staging-maven-plugin</artifactId>

               <version>1.6.7</version>

               <extensions>true</extensions>

               <configuration>

                   <serverId>ossrh</serverId>

                   <nexusUrl>https://oss.sonatype.org/</nexusUrl>

                   <autoReleaseAfterClose>true</autoReleaseAfterClose>

               </configuration>

           </plugin>

       </plugins>

   </build>

</project>

3、其他方案


https://github.com/rholder/guava-retrying


https://github.com/elennick/retry4j



————————————————

版权声明:本文为CSDN博主「明明如月学长」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。

原文链接:https://blog.csdn.net/w605283073/article/details/89038394

相关文章
|
4月前
|
Java
java基础(11)函数重载以及函数递归求和
Java支持函数重载,即在同一个类中可以声明多个同名方法,只要它们的参数类型和个数不同。函数重载与修饰符、返回值无关,但与参数的类型、个数、顺序有关。此外,文中还展示了如何使用递归方法`sum`来计算两个数之间的和,递归的终止条件是当第一个参数大于第二个参数时。
36 1
java基础(11)函数重载以及函数递归求和
|
3月前
|
Java
让星星⭐月亮告诉你,jdk1.8 Java函数式编程示例:Lambda函数/方法引用/4种内建函数式接口(功能性-/消费型/供给型/断言型)
本示例展示了Java中函数式接口的使用,包括自定义和内置的函数式接口。通过方法引用,实现对字符串操作如转换大写、数值转换等,并演示了Function、Consumer、Supplier及Predicate四种主要内置函数式接口的应用。
33 1
|
3月前
|
Java 编译器 C语言
【一步一步了解Java系列】:Java中的方法对标C语言中的函数
【一步一步了解Java系列】:Java中的方法对标C语言中的函数
33 3
|
5月前
|
Java 调度 Android开发
Android经典实战之Kotlin的delay函数和Java中的Thread.sleep有什么不同?
本文介绍了 Kotlin 中的 `delay` 函数与 Java 中 `Thread.sleep` 方法的区别。两者均可暂停代码执行,但 `delay` 适用于协程,非阻塞且高效;`Thread.sleep` 则阻塞当前线程。理解这些差异有助于提高程序效率与可读性。
89 1
|
6月前
|
存储 Java 编译器
Java中ArrayList的常用函数
确切地说,`ArrayList` 提供的这些方法构成了一套强大并且灵活的工具集,可以满足各种程序设计情况中的需求。例如,通过使用 `iterator()`方法,开发者可以在不知道集合大小的情况下遍历集合中全部或部分元素;而 `sort()`方法则能够对集合中的元素进行排序。这些函数在日常的Java编程中极其常见且重要,掌握它们对于进行集合操作和数据处理来说是基础且必须的。
44 2
Java中ArrayList的常用函数
|
5月前
|
存储 运维 Java
函数计算产品使用问题之怎么配置定时触发器来调用Java函数
函数计算产品作为一种事件驱动的全托管计算服务,让用户能够专注于业务逻辑的编写,而无需关心底层服务器的管理与运维。你可以有效地利用函数计算产品来支撑各类应用场景,从简单的数据处理到复杂的业务逻辑,实现快速、高效、低成本的云上部署与运维。以下是一些关于使用函数计算产品的合集和要点,帮助你更好地理解和应用这一服务。
|
5月前
|
开发框架 Java Android开发
JNI中调用Java函数
JNI中调用Java函数
37 0
|
5月前
|
开发框架 Java Android开发
JNI中调用Java函数
JNI中调用Java函数
42 0
|
6月前
|
Rust Cloud Native Java
Java演进问题之Serverless应用或函数的冷启动如何解决
Java演进问题之Serverless应用或函数的冷启动如何解决
|
7月前
|
存储 算法 搜索推荐
Java中的数组函数库及其使用技巧
Java中的数组函数库及其使用技巧