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

SpringBoot實現(xiàn)重試機制的四種方案

 更新時間:2025年04月14日 09:33:24   作者:風象南  
在分布式系統(tǒng)和微服務架構中,服務調(diào)用失敗是不可避免的現(xiàn)象,網(wǎng)絡不穩(wěn)定、服務過載、臨時故障等因素都可能導致調(diào)用失敗,重試機制作為一種處理臨時性故障的解決方案,能夠有效提高系統(tǒng)的可用性,需要的朋友可以參考下

一、Spring Retry

1. 基本原理

Spring Retry是Spring官方提供的重試框架,作為Spring生態(tài)系統(tǒng)的一部分,它通過AOP(面向切面編程)實現(xiàn)了對方法調(diào)用的重試能力。當方法調(diào)用失敗時,Spring Retry會根據(jù)配置的策略自動重新執(zhí)行該方法,直到成功或達到最大重試次數(shù)。

Spring Retry的核心組件包括:

  • RetryOperations:定義重試操作的接口
  • RetryTemplate:RetryOperations的默認實現(xiàn)
  • RetryPolicy:定義何時進行重試(如最大次數(shù)、重試的異常類型等)
  • BackOffPolicy:定義重試間隔策略(如固定間隔、指數(shù)退避等)
  • RecoveryCallback:定義最終失敗后的恢復策略

2. 集成配置

在SpringBoot項目中集成Spring Retry:

<dependency>
    <groupId>org.springframework.retry</groupId>
    <artifactId>spring-retry</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-aspects</artifactId>
</dependency>

然后在啟動類上啟用重試功能:

@SpringBootApplication
@EnableRetry
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }
}

3. 使用方法

Spring Retry提供了注解方式和編程方式兩種使用方法。

注解方式

@Service
public class RemoteServiceClient {
    
    private static final Logger logger = LoggerFactory.getLogger(RemoteServiceClient.class);
    
    @Retryable(value = {ServiceException.class}, 
               maxAttempts = 3, 
               backoff = @Backoff(delay = 1000, multiplier = 2))
    public String callRemoteService(String param) {
        logger.info("調(diào)用遠程服務,參數(shù): {}", param);
        // 模擬調(diào)用失敗
        if (Math.random() > 0.7) {
            logger.error("服務調(diào)用失敗");
            throw new ServiceException("遠程服務暫時不可用");
        }
        return "調(diào)用成功: " + param;
    }
    
    @Recover
    public String recover(ServiceException e, String param) {
        logger.warn("重試失敗,執(zhí)行恢復方法, 參數(shù): {}", param);
        return "降級響應: " + param;
    }
}

在上面的例子中:

  • @Retryable 注解定義了需要重試的方法,包括觸發(fā)重試的異常類型、最大重試次數(shù)和退避策略
  • backoff 屬性設置初始延遲1秒,且每次延遲時間翻倍(指數(shù)退避)
  • @Recover 注解定義了重試失敗后的恢復方法

編程方式

@Service
public class RemoteServiceClient {
    
    private final RetryTemplate retryTemplate;
    
    @Autowired
    public RemoteServiceClient(RetryTemplate retryTemplate) {
        this.retryTemplate = retryTemplate;
    }
    
    public String callWithRetry(String param) {
        return retryTemplate.execute(context -> {
            // 重試的業(yè)務邏輯
            if (Math.random() > 0.7) {
                throw new ServiceException("服務暫時不可用");
            }
            return "調(diào)用成功: " + param;
        }, context -> {
            // 重試失敗后的恢復邏輯
            return "降級響應: " + param;
        });
    }
    
    @Bean
    public RetryTemplate retryTemplate() {
        RetryTemplate template = new RetryTemplate();
        
        // 設置重試策略
        SimpleRetryPolicy policy = new SimpleRetryPolicy();
        policy.setMaxAttempts(3);
        
        // 設置退避策略
        ExponentialBackOffPolicy backOffPolicy = new ExponentialBackOffPolicy();
        backOffPolicy.setInitialInterval(1000);
        backOffPolicy.setMultiplier(2.0);
        
        template.setRetryPolicy(policy);
        template.setBackOffPolicy(backOffPolicy);
        
        return template;
    }
}

4. 優(yōu)缺點

優(yōu)點

  • 與Spring生態(tài)系統(tǒng)完美集成
  • 提供了豐富的重試策略和配置選項
  • 支持注解和編程兩種方式,使用靈活
  • 可以精確控制重試的異常類型
  • 支持聲明式事務回滾和提交

缺點

  • 依賴Spring框架
  • 代碼侵入性相對較強
  • 在復雜場景下配置略顯復雜
  • 與其他容錯機制集成需要額外工作

5. 適用場景

  • Spring生態(tài)系統(tǒng)的項目
  • 需要精細控制重試條件和策略的場景
  • 與Spring事務結合的業(yè)務場景
  • 方法級別的重試需求

二、Resilience4j Retry

1. 基本原理

Resilience4j是受Netflix Hystrix啟發(fā)而創(chuàng)建的輕量級容錯庫,其中Resilience4j Retry模塊提供了強大的重試功能。與Spring Retry不同,Resilience4j采用函數(shù)式編程風格,使用裝飾器模式實現(xiàn)重試功能。

Resilience4j Retry的特點:

  • 基于函數(shù)式接口
  • 無外部依賴,輕量級設計
  • 可與其他容錯機制(如斷路器、限流器)無縫集成
  • 提供豐富的監(jiān)控指標

2. 集成配置

在SpringBoot項目中集成Resilience4j Retry:

<dependency>
    <groupId>io.github.resilience4j</groupId>
    <artifactId>resilience4j-spring-boot2</artifactId>
    <version>1.7.0</version>
</dependency>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-aop</artifactId>
</dependency>

配置application.yml:

resilience4j.retry:
  instances:
    backendService:
      maxAttempts: 3
      waitDuration: 1s
      enableExponentialBackoff: true
      exponentialBackoffMultiplier: 2
      retryExceptions:
        - java.io.IOException
        - java.util.concurrent.TimeoutException

3. 使用方法

Resilience4j同樣支持注解方式和編程方式。

注解方式

@Service
public class BackendService {
    
    private static final Logger logger = LoggerFactory.getLogger(BackendService.class);
    
    @Retry(name = "backendService", fallbackMethod = "fallbackCall")
    public String callBackendService(String param) {
        logger.info("調(diào)用后端服務,參數(shù): {}", param);
        
        if (Math.random() > 0.7) {
            logger.error("服務調(diào)用失敗");
            throw new IOException("服務連接失敗");
        }
        
        return "后端服務響應: " + param;
    }
    
    public String fallbackCall(String param, Exception ex) {
        logger.warn("所有重試失敗,執(zhí)行降級方法,參數(shù): {}", param);
        return "降級響應: " + param;
    }
}

編程方式

@Service
public class BackendService {
    
    private final RetryRegistry retryRegistry;
    private final Logger logger = LoggerFactory.getLogger(BackendService.class);
    
    @Autowired
    public BackendService(RetryRegistry retryRegistry) {
        this.retryRegistry = retryRegistry;
    }
    
    public String executeWithRetry(String param) {
        // 獲取已配置的重試實例
        Retry retry = retryRegistry.retry("backendService");
        
        // 創(chuàng)建一個可重試的函數(shù)
        CheckedFunction0<String> retryableFunction = Retry.decorateCheckedSupplier(
            retry, () -> callBackendService(param));
        
        try {
            // 執(zhí)行重試函數(shù)
            return retryableFunction.apply();
        } catch (Throwable throwable) {
            logger.error("重試失敗: {}", throwable.getMessage());
            return "降級響應: " + param;
        }
    }
    
    private String callBackendService(String param) throws IOException {
        logger.info("調(diào)用后端服務,參數(shù): {}", param);
        
        if (Math.random() > 0.7) {
            throw new IOException("服務連接失敗");
        }
        
        return "后端服務響應: " + param;
    }
}

4. 優(yōu)缺點

優(yōu)點

  • 輕量級設計,無外部依賴
  • 函數(shù)式編程風格,代碼簡潔
  • 提供豐富的監(jiān)控和統(tǒng)計指標
  • 可與斷路器、限流器等容錯機制無縫集成
  • 支持多種高級重試策略

缺點

  • 學習曲線相對陡峭,尤其是函數(shù)式概念
  • 對于不熟悉函數(shù)式編程的開發(fā)者可能不夠直觀
  • 某些高級功能需要額外配置

5. 適用場景

  • 需要與其他容錯機制結合的復雜場景
  • 微服務架構中的服務間調(diào)用
  • 需要詳細監(jiān)控指標的系統(tǒng)

三、Guava Retrying

1. 基本原理

Guava Retrying是Google Guava庫提供的重試機制,它提供了一個簡單靈活的API來實現(xiàn)重試功能。

Guava Retrying通過構建器模式提供了靈活的重試配置,可以自定義重試條件、停止策略、等待策略等。

2. 集成配置

在SpringBoot項目中集成Guava Retrying:

<dependency>
    <groupId>com.github.rholder</groupId>
    <artifactId>guava-retrying</artifactId>
    <version>2.0.0</version>
</dependency>

3. 使用方法

Guava Retrying主要采用編程方式使用:

@Service
public class ExternalServiceClient {
    
    private static final Logger logger = LoggerFactory.getLogger(ExternalServiceClient.class);
    
    public String callExternalService(String param) {
        Retryer<String> retryer = RetryerBuilder.<String>newBuilder()
            .retryIfException() // 發(fā)生任何異常時重試
            .retryIfResult(result -> result == null) // 結果為null時重試
            .withWaitStrategy(WaitStrategies.exponentialWait(1000, 10000, TimeUnit.MILLISECONDS)) // 指數(shù)退避
            .withStopStrategy(StopStrategies.stopAfterAttempt(3)) // 最多重試3次
            .withRetryListener(new RetryListener() {
                @Override
                public <V> void onRetry(Attempt<V> attempt) {
                    logger.info("第{}次重試", attempt.getAttemptNumber());
                    if (attempt.hasException()) {
                        logger.error("異常: {}", attempt.getExceptionCause().getMessage());
                    }
                }
            })
            .build();
        
        try {
            return retryer.call(() -> {
                logger.info("調(diào)用外部服務,參數(shù): {}", param);
                
                // 模擬服務調(diào)用
                if (Math.random() > 0.7) {
                    throw new RuntimeException("服務暫時不可用");
                }
                
                return "外部服務響應: " + param;
            });
        } catch (RetryException | ExecutionException e) {
            logger.error("重試失敗: {}", e.getMessage());
            return "降級響應: " + param;
        }
    }
}

在SpringBoot中創(chuàng)建可復用的Retryer bean:

@Configuration
public class RetryConfig {
    
    @Bean
    public <T> Retryer<T> defaultRetryer() {
        return RetryerBuilder.<T>newBuilder()
            .retryIfException()
            .retryIfResult(Predicates.isNull())
            .withWaitStrategy(WaitStrategies.exponentialWait(100, 1000, TimeUnit.MILLISECONDS))
            .withStopStrategy(StopStrategies.stopAfterAttempt(3))
            .build();
    }
}

@Service
public class ServiceWithRetry {
    
    private final Retryer<String> retryer;
    
    @Autowired
    public ServiceWithRetry(Retryer<String> retryer) {
        this.retryer = retryer;
    }
    
    public String executeWithRetry(String input) throws ExecutionException, RetryException {
        return retryer.call(() -> {
            // 業(yè)務邏輯
            return processInput(input);
        });
    }
}

4. 高級特性

Guava Retrying提供了豐富的定制選項:

Retryer<String> complexRetryer = RetryerBuilder.<String>newBuilder()
    // 定制重試條件
    .retryIfExceptionOfType(IOException.class)
    .retryIfException(e -> e instanceof TimeoutException)
    .retryIfResult(result -> result != null && result.contains("error"))
    
    // 定制等待策略
    .withWaitStrategy(WaitStrategies.join(
        WaitStrategies.fixedWait(1000, TimeUnit.MILLISECONDS),
        WaitStrategies.randomWait(1000, TimeUnit.MILLISECONDS, 2000, TimeUnit.MILLISECONDS)
    ))
    
    // 定制停止策略
    .withStopStrategy(StopStrategies.stopAfterDelay(30, TimeUnit.SECONDS))
    
    // 定制阻塞策略
    .withBlockStrategy(BlockStrategies.threadSleepStrategy())
    
    .build();

5. 優(yōu)缺點

優(yōu)點

  • API簡單直觀,容易上手
  • 高度可定制的重試條件、等待策略和停止策略
  • 不依賴Spring框架,可在任何Java項目中使用

缺點

  • 沒有注解支持,只能通過編程方式使用
  • 缺乏與Spring生態(tài)系統(tǒng)的深度集成
  • 沒有內(nèi)置的監(jiān)控和統(tǒng)計功能
  • 已停止更新

6. 適用場景

  • 簡單的重試需求
  • 非Spring項目或?qū)pring依賴較少的項目
  • 需要高度自定義重試邏輯的場景

四、Failsafe

1. 基本原理

failsafe是一個相對較新的Java重試庫,專注于高性能和低延遲場景。它的設計目標是提供一個簡單、高效的重試機制,同時保持API的簡潔性和使用的便捷性。failsafe支持同步和異步重試,具有靈活的重試策略和最小的依賴。

2. 集成配置

在SpringBoot項目中集成failsafe

<dependency>
    <groupId>dev.failsafe</groupId>
    <artifactId>failsafe</artifactId>
    <version>3.3.2</version>
</dependency>

failsafe通常通過dev.failsafe:failsafe庫來使用,這是一個現(xiàn)代化的重試和容錯庫。

3. 使用方法

failsafe主要采用編程方式使用,具有流式API設計

@Service
public class FailsafeService {
    
    private static final Logger logger = LoggerFactory.getLogger(FailsafeService.class);
    
    public String executeWithRetry(String param) {
        return Failsafe.with(
            // 配置重試策略
            RetryPolicy.<String>builder()
                .handle(IOException.class, TimeoutException.class)
                .withMaxRetries(3)
                .withDelay(Duration.ofSeconds(1))
                .withMaxDuration(Duration.ofSeconds(10))
                .withBackoff(Duration.ofMillis(100), Duration.ofSeconds(2))
                .onRetry(event -> logger.info("第{}次重試,上次異常: {}", 
                                         event.getAttemptCount(), 
                                         event.getLastException().getMessage()))
                .onFailure(event -> logger.error("重試失敗,嘗試次數(shù): {}, 總耗時: {}ms", 
                                            event.getAttemptCount(),
                                            event.getElapsedTime().toMillis()))
                .build()
        )
        .get(() -> {
            logger.info("執(zhí)行操作,參數(shù): {}", param);
            
            // 模擬操作
            if (Math.random() > 0.7) {
                throw new IOException("操作暫時失敗");
            }
            
            return "操作成功: " + param;
        });
    }
    
    // 異步重試示例
    public CompletableFuture<String> executeWithRetryAsync(String param) {
        return Failsafe.with(
            RetryPolicy.<String>builder()
                .handle(IOException.class)
                .withMaxRetries(3)
                .withBackoff(Duration.ofMillis(100), Duration.ofSeconds(1))
                .build()
        )
        .getAsync(() -> {
            logger.info("異步執(zhí)行操作,參數(shù): {}", param);
            
            // 模擬異步操作
            if (Math.random() > 0.7) {
                throw new IOException("異步操作暫時失敗");
            }
            
            return "異步操作成功: " + param;
        });
    }
    
    // 帶降級的重試示例
    public String executeWithFallback(String param) {
        return Failsafe.with(
            RetryPolicy.<String>builder()
                .handle(IOException.class)
                .withMaxRetries(3)
                .build(),
            // 降級策略
            Fallback.of(e -> "降級響應: " + param)
        )
        .get(() -> {
            // 業(yè)務邏輯
            if (Math.random() > 0.7) {
                throw new IOException("操作失敗");
            }
            return "操作成功: " + param;
        });
    }
}

在SpringBoot中創(chuàng)建可復用的RetryPolicy bean

@Configuration
public class FailsafeConfig {
    
    @Bean
    public RetryPolicy<Object> defaultRetryPolicy() {
        return RetryPolicy.builder()
            .handle(Exception.class)
            .withMaxRetries(3)
            .withBackoff(Duration.ofMillis(100), Duration.ofSeconds(1), 2.0)
            .build();
    }
    
    @Bean
    public Fallback<Object> defaultFallback() {
        return Fallback.of(e -> {
            if (e instanceof ServiceException) {
                return "服務異常降級";
            }
            return "通用降級響應";
        });
    }
}

@Service
public class ServiceWithFailsafeRetry {
    
    private final RetryPolicy<Object> defaultRetryPolicy;
    private final Fallback<Object> defaultFallback;
    
    @Autowired
    public ServiceWithFailsafeRetry(RetryPolicy<Object> defaultRetryPolicy, 
                               Fallback<Object> defaultFallback) {
        this.defaultRetryPolicy = defaultRetryPolicy;
        this.defaultFallback = defaultFallback;
    }
    
    public String executeWithRetry(String input) {
        return Failsafe.with(defaultRetryPolicy, defaultFallback)
            .get(() -> {
                // 業(yè)務邏輯
                return processInput(input);
            });
    }
}

4. 優(yōu)缺點

優(yōu)點

  • 極高的性能,適合高頻調(diào)用場景
  • 支持同步和異步重試
  • 輕量級,依賴少
  • 與CompletableFuture良好集成
  • 內(nèi)置豐富的監(jiān)聽器機制

缺點

  • 沒有注解支持,只能通過編程方式使用
  • 與Spring框架集成度不高
  • 近幾年更新也不活躍

5. 適用場景

  • 高性能、低延遲要求的應用
  • 需要異步重試能力的場景
  • 需要細粒度控制重試行為的場景

五、四種重試機制的對比

特性Spring RetryResilience4j RetryGuava RetryingFailsafe
編程模型AOP + 命令式函數(shù)式命令式流式
注解支持支持支持不支持不支持
依賴Spring無外部依賴Guava最小依賴
性能開銷中等中等極低
異步支持有限良好有限優(yōu)秀
監(jiān)控集成有限豐富基本
配置方式注解/編程配置文件/注解/編程編程編程
與其他容錯機制集成有限原生支持良好
學習曲線中等較陡平緩平緩
可定制性
適用場景Spring項目微服務/云原生應用簡單場景/非Spring項目高性能場景

六、最佳實踐與注意事項

1. 通用最佳實踐

  • 確保冪等性:重試機制最適合用于冪等操作,即多次執(zhí)行產(chǎn)生相同結果的操作。對于非冪等操作,需要特別小心。
  • 設置合理的超時和重試次數(shù):避免無限重試或重試時間過長,通常3-5次足夠處理大多數(shù)臨時故障。
  • 使用指數(shù)退避策略:隨著重試次數(shù)增加,逐漸增加重試間隔,避免對目標服務造成過大壓力。
  • 區(qū)分臨時故障和永久故障:只對可能自行恢復的臨時故障進行重試,對于永久性錯誤不應重試。
  • 添加監(jiān)控和日志:記錄重試次數(shù)、成功率等指標,便于問題排查和性能優(yōu)化。

2. 避免常見陷阱

  • 重試風暴:當多個客戶端同時對一個故障服務進行重試時,可能導致服務負載激增。
  • 資源泄漏:重試過程中要確保資源(如數(shù)據(jù)庫連接、HTTP連接)正確釋放。
  • 過度重試:過度重試可能導致性能下降,應設置合理的最大重試次數(shù)和總超時時間。
  • 重試成本:某些操作重試成本高昂(如涉及第三方付費API),需謹慎設計重試策略。

七、總結

選擇合適的重試機制應基于項目的技術棧、復雜度和需求。無論選擇哪種機制,都應遵循重試的最佳實踐,避免常見陷阱,確保系統(tǒng)的穩(wěn)定性和可靠性。

以上就是SpringBoot實現(xiàn)重試機制的四種方案的詳細內(nèi)容,更多關于SpringBoot重試機制的資料請關注腳本之家其它相關文章!

相關文章

  • java?for循環(huán)內(nèi)執(zhí)行多線程問題

    java?for循環(huán)內(nèi)執(zhí)行多線程問題

    這篇文章主要介紹了java?for循環(huán)內(nèi)執(zhí)行多線程問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-03-03
  • SpringBoot使用Redis的zset統(tǒng)計在線用戶信息

    SpringBoot使用Redis的zset統(tǒng)計在線用戶信息

    這篇文章主要介紹了SpringBoot使用Redis的zset統(tǒng)計在線用戶信息,幫助大家更好的理解和學習使用SpringBoot框架,感興趣的朋友可以了解下
    2021-04-04
  • Spring?Boot?整合持久層之MyBatis

    Spring?Boot?整合持久層之MyBatis

    在實際開發(fā)中不僅僅是要展示數(shù)據(jù),還要構成數(shù)據(jù)模型添加數(shù)據(jù),這篇文章主要介紹了SpringBoot集成Mybatis操作數(shù)據(jù)庫,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2022-08-08
  • Java向List集合中批量添加元素的實現(xiàn)方法

    Java向List集合中批量添加元素的實現(xiàn)方法

    這篇文章主要介紹了Java向List集合中批量添加元素的實現(xiàn)方法,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-05-05
  • spring的13個經(jīng)典面試題

    spring的13個經(jīng)典面試題

    Spring框架是一個開放源代碼的J2EE應用程序框架,是針對bean的生命周期進行管理的輕量級容Spring解決了開發(fā)者在J2EE開發(fā)中遇到的許多常見的問題,我們這篇文章就來了解一下spring的面試題
    2021-06-06
  • Java設計模式之單態(tài)模式(Singleton模式)介紹

    Java設計模式之單態(tài)模式(Singleton模式)介紹

    這篇文章主要介紹了Java設計模式之單態(tài)模式(Singleton模式)介紹,本文講解了如何使用單例模式、使用單例模式注意事項等內(nèi)容,需要的朋友可以參考下
    2015-03-03
  • Spring?boot整合jsp和tiles模板示例

    Spring?boot整合jsp和tiles模板示例

    這篇文章主要介紹了Spring?boot整合jsp模板和tiles模板的示例演示過程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-03-03
  • 解析Java的Spring框架的BeanPostProcessor發(fā)布處理器

    解析Java的Spring框架的BeanPostProcessor發(fā)布處理器

    這篇文章主要介紹了Java的Spring框架的BeanPostProcessor發(fā)布處理器,Spring是Java的SSH三大web開發(fā)框架之一,需要的朋友可以參考下
    2015-12-12
  • Java 在PDF中添加條形碼的兩種方法

    Java 在PDF中添加條形碼的兩種方法

    本文就將通過使用Java程序來演示如何在PDF文檔中添加Codebar、Code128A和Code39條形碼。通過實例代碼給大家講解的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友參考下吧
    2020-03-03
  • Java?數(shù)據(jù)結構深入理解ArrayList與順序表

    Java?數(shù)據(jù)結構深入理解ArrayList與順序表

    ArrayList?類是一個可以動態(tài)修改的數(shù)組,與普通數(shù)組的區(qū)別就是它是沒有固定大小的限制,我們可以添加或刪除元素。ArrayList?繼承了?AbstractList?,并實現(xiàn)了?List?接口,順序表是將元素順序地存放在一塊連續(xù)的存儲區(qū)里,元素間的順序關系由它們的存儲順序自然表示
    2022-04-04

最新評論