欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java實現(xiàn)自定義重試工具類

 更新時間:2024年11月29日 11:29:08   作者:花開不識君  
這篇文章主要為大家詳細介紹了如何基于Java實現(xiàn)自定義重試工具類,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下

Spring-retry、guava的Retry都提供有重試工具,但二者均存在一個確缺點,即如果重試等待過程中會一直阻塞工作線程,這對于在生產(chǎn)環(huán)境使用是存在風險的,如果存在大量長時間等待的重試任務將會耗盡系統(tǒng)線程資源,下文基于線程池來完成一個簡易的重試工具類。

核心思想

將任務封裝為一個task,將任務的重試放入可調(diào)度的線程池中完成執(zhí)行,避免在重試間隔中,線程陷入無意義的等待,同時將重試機制抽象為重試策略。

代碼實現(xiàn)

重試工具類

package com.huakai.springenv.retry.v2;

import lombok.extern.slf4j.Slf4j;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.Function;

@Slf4j
public class RetryUtil {

    public static ExecutorService EXECUTOR = Executors.newFixedThreadPool(1);
    private static final ScheduledExecutorService SCHEDULER_EXECUTOR = Executors.newScheduledThreadPool(20);


    /**
     * 任務重試
     * @param actualTaskFunction 執(zhí)行的任務函數(shù)
     * @param resultHandler 任務結果處理器
     * @param maxRetry 最大重試次數(shù)
     * @param retryStrategy 重試策略
     */
    public static void retryTask(
            Function<Integer, String> actualTaskFunction,
            Function<String, Boolean> resultHandler,
            int maxRetry,
            RetryStrategy retryStrategy // 使用策略模式
    ) {
        Runnable runnable = new Runnable() {
            final AtomicInteger retryCount = new AtomicInteger(); // 當前重試次數(shù)
            final AtomicInteger maxRetryCount = new AtomicInteger(maxRetry); // 最大重試次數(shù)

            @Override
            public void run() {
                String taskResult = actualTaskFunction.apply(retryCount.get()); // 執(zhí)行任務
                Boolean taskSuccess = resultHandler.apply(taskResult); // 處理任務結果
                if (taskSuccess) {
                    if (retryCount.get() > 1) {
                        log.info("任務重試成功,重試次數(shù):{}", retryCount.get());
                    }
                    return; // 任務成功,不需要再重試
                }

                if (retryCount.incrementAndGet() == maxRetryCount.get()) {
                    log.warn("任務重試失敗,重試次數(shù):{}", retryCount.get());
                    return; // 達到最大重試次數(shù),停止重試
                }

                // 獲取重試間隔
                long delay = retryStrategy.getDelay(retryCount.get());
                TimeUnit timeUnit = retryStrategy.getTimeUnit(retryCount.get());

                // 安排下次重試
                SCHEDULER_EXECUTOR.schedule(this, delay, timeUnit);
                log.info("任務重試失敗,等待 {} {} 后再次嘗試,當前重試次數(shù):{}", delay, timeUnit, retryCount.get());
            }
        };
        EXECUTOR.execute(runnable); // 執(zhí)行任務
    }

    public static void main(String[] args) {
        // 使用指數(shù)退避重試策略
        RetryStrategy retryStrategy = new ExponentialBackoffRetryStrategy(1, TimeUnit.SECONDS);

        retryTask(
                retryCount -> "task result",
                taskResult -> Math.random() < 0.1,
                5,
                retryStrategy
        );
    }
}

重試策略

指數(shù)退避

package com.huakai.springenv.retry.v2;

import java.util.concurrent.TimeUnit;

/**
 * 指數(shù)退避重試策略
 */
public class ExponentialBackoffRetryStrategy implements RetryStrategy {
    private final long initialDelay;
    private final TimeUnit timeUnit;

    public ExponentialBackoffRetryStrategy(long initialDelay, TimeUnit timeUnit) {
        this.initialDelay = initialDelay;
        this.timeUnit = timeUnit;
    }

    @Override
    public long getDelay(int retryCount) {
        return (long) (initialDelay * Math.pow(2, retryCount - 1)); // 指數(shù)退避
    }

    @Override
    public TimeUnit getTimeUnit(int retryCount) {
        return timeUnit;
    }
}

自定義重試間隔時間

package com.huakai.springenv.retry.v2;

import java.util.List;
import java.util.concurrent.TimeUnit;

/**
 * 自定義重試間隔時間的重試策略
 */
public class CustomerIntervalRetryStrategy implements RetryStrategy {
    // 配置重試間隔和時間單位
    List<RetryInterval> retryIntervals;


    public CustomerIntervalRetryStrategy(List<RetryInterval> retryIntervals) {
        this.retryIntervals = retryIntervals;
    }

    @Override
    public long getDelay(int retryCount) {
        return retryIntervals.get(retryCount).getDelay();
    }

    @Override
    public TimeUnit getTimeUnit(int retryCount){
        return retryIntervals.get(retryCount).getTimeUnit();
    }
}

固定間隔

package com.huakai.springenv.retry.v2;

import java.util.concurrent.TimeUnit;

/**
 * 固定間隔重試策略
 */
public class FixedIntervalRetryStrategy implements RetryStrategy {
    private final long interval;
    private final TimeUnit timeUnit;

    public FixedIntervalRetryStrategy(long interval, TimeUnit timeUnit) {
        this.interval = interval;
        this.timeUnit = timeUnit;
    }

    @Override
    public long getDelay(int retryCount) {
        return interval;
    }

    @Override
    public TimeUnit getTimeUnit(int retryCount) {
        return timeUnit;
    }
}

到此這篇關于Java實現(xiàn)自定義重試工具類的文章就介紹到這了,更多相關Java重試工具類內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • MyBatis使用Map與模糊查詢的方法示例

    MyBatis使用Map與模糊查詢的方法示例

    這篇文章主要給大家介紹了關于MyBatis使用Map與模糊查詢的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-05-05
  • Java中的弗洛伊德(Floyd)算法

    Java中的弗洛伊德(Floyd)算法

    這篇文章主要介紹了Java中的弗洛伊德(Floyd)算法,Floyd算法又稱為插點法,是一種利用動態(tài)規(guī)劃的思想尋找給定的加權圖中多源點之間最短路徑的算法,與Dijkstra算法類似,需要的朋友可以參考下
    2024-01-01
  • MyBatis結果映射(ResultMap)的使用

    MyBatis結果映射(ResultMap)的使用

    在MyBatis中,結果映射是實現(xiàn)數(shù)據(jù)庫結果集到Java對象映射的核心,它不僅支持簡單的字段映射,還能處理字段名不一致、嵌套對象和集合映射等復雜場景,通過ResultMap,開發(fā)者可以靈活定義映射關系,以適應各種需求,感興趣的可以了解一下
    2024-09-09
  • Scala中優(yōu)雅的處理Null問題

    Scala中優(yōu)雅的處理Null問題

    Spark 采用混合方式,大部分情況下使用 Option,但個別時候出于性能原因才使用了null。一個很好的習慣是當有方法返回值可能為null的時候,使用Option來代替,本文給大家介紹Scala處理Null的知識詳解,一起看看吧
    2021-08-08
  • java中幾種http請求方式示例詳解

    java中幾種http請求方式示例詳解

    在日常工作和學習中有很多地方都需要發(fā)送HTTP請求,下面這篇文章主要給大家介紹了關于java中幾種http請求方式的相關資料,文中通過代碼介紹的非常詳細,需要的朋友可以參考下
    2023-11-11
  • 淺談java中字節(jié)與字符的區(qū)別

    淺談java中字節(jié)與字符的區(qū)別

    這篇文章主要介紹了淺談java中字節(jié)與字符的區(qū)別,字節(jié)是java中的基本數(shù)據(jù)類型,用來申明字節(jié)型的變量;字符是語義上的單位,它是有編碼的,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-07-07
  • 簡單談談Spring Ioc原理解析

    簡單談談Spring Ioc原理解析

    學習過Spring框架的人一定都會聽過Spring的IoC(控制反轉) 、DI(依賴注入)這兩個概念,對于初學Spring的人來說,總覺得IoC 、DI這兩個概念是模糊不清的,是很難理解的,今天和大家分享網(wǎng)上的一些技術大牛們對Spring框架的IOC的理解以及談談我對Spring Ioc的理解。
    2018-09-09
  • 詳解Maven私服Nexus的安裝與使用

    詳解Maven私服Nexus的安裝與使用

    這篇文章主要介紹了詳解Maven私服Nexus的安裝與使用,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-03-03
  • 淺析打開eclipse出現(xiàn)Incompatible JVM的解決方法

    淺析打開eclipse出現(xiàn)Incompatible JVM的解決方法

    本篇文章是對打開eclipse出現(xiàn)Incompatible JVM的解決方法進行了詳細的分析介紹,需要的朋友參考下
    2013-07-07
  • 深入理解Java中觀察者模式與委托的對比

    深入理解Java中觀察者模式與委托的對比

    這篇文章主要介紹了Java中觀察者模式與委托的對比,觀察者模式:定義了一種一對多的依賴關系,讓多個觀察者對象同時監(jiān)聽某一個主題對象,委托的實現(xiàn)簡單來講就是用反射來實現(xiàn)的,本文給大家介紹的非常詳細,需要的朋友可以參考下
    2022-05-05

最新評論