使用 Spring @Retryable 实现自动重试机制

在微服务架构中,服务之间的调用可能会因为一些暂时性的错误而失败,例如网络波动、数据库连接超时或第三方服务不可用等。为了提高系统的可靠性和容错性,我们可以使用自动重试机制来应对这些临时故障。Spring 提供了 @Retryable 注解来方便地实现这一功能。

在本文中,我们将介绍如何在 Spring 中使用 @Retryable 实现自动重试机制,帮助你轻松应对常见的服务调用失败问题。

1. 什么是 @Retryable

@Retryable 是 Spring Retry 提供的注解,它允许我们在方法执行失败时自动进行重试。你可以指定重试的次数、重试的间隔时间,以及触发重试的异常类型。

通过 @Retryable,你可以大大简化异常处理逻辑,将重试的复杂性交给 Spring Retry 处理,避免了手动实现重试机制的繁琐。

2. 如何在 Spring 中使用 @Retryable
2.1 添加依赖

首先,确保你的项目中包含 Spring Retry 相关的依赖。如果你正在使用 Spring Boot,添加如下依赖:

<dependency>
    <groupId>org.springframework.retry</groupId>
    <artifactId>spring-retry</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-aop</artifactId>
</dependency>

然后在 Spring Boot 的主类或配置类中启用 Spring Retry 功能:

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

@EnableRetry 注解启用 Spring Retry 功能,它会自动为被标记为 @Retryable 的方法提供重试机制。

2.2 使用 @Retryable 注解

在需要重试的业务方法上使用 @Retryable 注解,指定触发重试的条件,如异常类型、最大重试次数和重试间隔等。

import org.springframework.retry.annotation.Retryable;
import org.springframework.stereotype.Service;

@Service
public class MyService {

    @Retryable(value = {Exception.class}, maxAttempts = 3, backoff = @Backoff(delay = 1000))
    public void performTask() throws Exception {
        System.out.println("Attempting to perform task...");
        // 模拟业务失败
        if (Math.random() > 0.5) {
            throw new Exception("Task failed");
        }
        System.out.println("Task completed successfully");
    }
}

上面的代码中:

  • value = {Exception.class}:指定哪些异常会触发重试。在这里,Exception 或其子类的异常会触发重试。
  • maxAttempts = 3:最大重试次数,包括第一次调用。如果方法抛出异常且重试次数未达到最大值,Spring Retry 会继续尝试。
  • backoff = @Backoff(delay = 1000):每次重试之间的延迟时间为 1000 毫秒(1 秒)。这有助于避免短时间内的多次重试带来过大的负载。
2.3 配置重试策略

除了 maxAttemptsbackoff@Retryable 还提供了一些其他选项来灵活配置重试策略。

  • backoff:指定重试的回退策略,可以设置:
    • delay:每次重试之间的间隔时间(单位:毫秒)。
    • multiplier:指数退避的乘数。
    • maxDelay:最大延迟时间。

示例:

@Retryable(value = {Exception.class}, maxAttempts = 3, backoff = @Backoff(delay = 1000, multiplier = 2, maxDelay = 5000))
public void performTask() throws Exception {
    // 执行业务逻辑
}

在这个例子中,第一次重试会间隔 1 秒,第二次重试会间隔 2 秒,第三次重试会间隔 4 秒,但不超过 5 秒。

2.4 恢复方法

你还可以定义一个恢复方法,在最大重试次数耗尽后执行。恢复方法可以用于处理最终的失败情况,避免系统崩溃。恢复方法的参数应该与重试方法的异常类型一致。

import org.springframework.retry.annotation.Backoff;
import org.springframework.retry.annotation.Retryable;
import org.springframework.retry.annotation.Recover;
import org.springframework.stereotype.Service;

@Service
public class MyService {

    @Retryable(value = {Exception.class}, maxAttempts = 3, backoff = @Backoff(delay = 1000))
    public void performTask() throws Exception {
        System.out.println("Attempting to perform task...");
        if (Math.random() > 0.5) {
            throw new Exception("Task failed");
        }
        System.out.println("Task completed successfully");
    }

    @Recover
    public void recover(Exception e) {
        System.out.println("Recovery from failure: " + e.getMessage());
    }
}

在这个例子中,如果 performTask 方法在尝试了 3 次后仍然失败,recover 方法将被调用,并传递失败的异常。

3. 典型应用场景
  • 临时的网络故障:例如,在调用远程 API 或微服务时可能会遇到短暂的网络问题。通过 @Retryable 注解,我们可以在网络问题恢复时自动进行重试,提升系统的可靠性。

  • 数据库连接超时:在进行数据库操作时,偶尔可能会遇到连接超时的问题。通过配置重试策略,可以在数据库连接恢复后重新尝试执行操作。

  • 消息队列消费失败:当消费消息时,如果发生暂时性故障(如消息处理超时、网络不通等),我们可以通过 @Retryable 实现自动重试,直到处理成功。

4. 注意事项
  • 重试次数要合理配置:重试次数过多可能导致系统压力增大,甚至引发其他问题。务必根据业务需求合理设置 maxAttempts

  • 避免过频繁的重试:如果重试间隔过短,可能会导致对系统造成过大负载。合理配置 backoff 参数,以避免频繁重试带来的资源消耗。

  • 恢复方法的使用:在重试次数耗尽后,使用恢复方法可以确保程序不会直接崩溃,可以做一些清理或后续处理。

5. 总结

使用 Spring @Retryable 注解可以非常方便地实现自动重试机制,帮助我们在面对临时故障时自动恢复,减少了手动处理失败的复杂性。通过合理配置重试次数、重试间隔和恢复方法,能够有效提高系统的容错能力和稳定性。无论是在网络请求、数据库操作,还是消息队列消费等场景中,@Retryable 都是一个非常有用的工具。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/948727.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

小程序开发全解析 快速构建高效应用的核心指南

内容概要 小程序开发是当前数字世界中炙手可热的领域&#xff0c;吸引了无数开发者和企业的关注。随着技术的进步&#xff0c;小程序成为了提升用户体验、增强品牌曝光以及增加客户互动的重要工具。了解小程序的基本概念&#xff0c;就像是打开了一扇通往新世界的大门。 在这…

SQL—Group_Concat函数用法详解

SQL—Group_Concat函数用法详解 在LC遇见的一道很有趣的SQL题&#xff0c;有用到这个函数&#xff0c;就借这道题抛砖引玉&#xff0c;在此讲解一下group_concat函数的用法。&#x1f923; GROUP_CONCAT([DISTINCT] expression [ORDER BY expression] [SEPARATOR separator])…

Edge Scdn的应用场景有哪些?

酷盾安全Edge Scdn 具备强大的安全防护能力&#xff0c;通过多层防御机制&#xff0c;如防火墙、DDoS 攻击防护、入侵检测和防御、数据加密等&#xff0c;有效抵御各种网络攻击&#xff0c;包括 DDoS 攻击、CC 攻击、SQL 注入攻击、XSS 跨站脚本攻击等&#xff0c;保障网站和应…

流光效果

1、流光效果是什么 在 Unity Shader 中的流光效果是一种动态的视觉效果&#xff0c;通常用于给材质增加一种闪光或光线移动的效果&#xff0c;使物体表面看起来像是有光在流动。这种效果常用于武器光效、能量护盾、传送门等等&#xff0c;可以让物体看起来更加生动富有科技感 …

滑动窗口——串联所有单词的子串

一.题目描述 30. 串联所有单词的子串 - 力扣&#xff08;LeetCode&#xff09; 二.题目解析 题目前提&#xff1a;s是一个字符串&#xff0c;words是一个字符串数组&#xff0c;里面所有的字符串的长度都是相等的。 题目要求&#xff1a;找到s中的一段连续的子串&#xff0…

【微软,模型规模】模型参数规模泄露:理解大型语言模型的参数量级

模型参数规模泄露&#xff1a;理解大型语言模型的参数量级 关键词&#xff1a; #大型语言模型 Large Language Model #参数规模 Parameter Scale #GPT-4o #GPT-4o-mini #Claude 3.5 Sonnet 具体实例与推演 近日&#xff0c;微软在一篇医学相关论文中意外泄露了OpenAI及Claud…

SpringBoot Maven 项目 pom 中的 plugin 插件用法整理

把 SpringBoot Maven 项目打包成 jar 文件时&#xff0c;我们通常用到 spring-boot-maven-plugin 插件。 前面也介绍过&#xff0c;在 spring-boot-starter-parent POM 和 spring-boot-starter POM 中都有插件的管理&#xff0c;现在我们就撸一把构建元素中插件的用法。 一、…

UE5AI感知组件

官方解释&#xff1a; AI感知系统为Pawn提供了一种从环境中接收数据的方式&#xff0c;例如噪音的来源、AI是否遭到破坏、或AI是否看到了什么。 AI感知组件&#xff08;AIPerception Component&#xff09;是用于实现游戏中的非玩家角色&#xff08;NPC&#xff09;对环境和其…

【数据仓库】hive on Tez配置

hive on Tez 搭建 前提是hive4.0hadoop3.2.2数仓已搭建完成&#xff0c;现在只是更换其执行引擎 为Tez。搭建可参考【数据仓库】hive hadoop数仓搭建实践文章。 Tez 下载 下载地址 https://archive.apache.org/dist/tez/ 官网地址 https://tez.apache.org/releases/apac…

finereport动态数据源插件教程2

场景&#xff1a; 模板中有多个数据集&#xff0c;只需要其中一个数据集按照不同的参数显示不同数据库的数据。 模板制作&#xff1a; 两个数据集ds1&#xff0c;ds2&#xff0c;ds1的绑定到参数面板的下拉框上&#xff0c;ds2显示到模板正文中&#xff0c;现在需要ds1根据不同…

Java通过谷歌邮箱Gmail直接发送邮件的三种方式

错误 Connected to the target VM, address: 127.0.0.1:52082, transport: socketException in thread "main" javax.mail.MessagingException: Got bad greeting from SMTP host: smtp.gmail.com, port: 587, response: [EOF] at com.sun.mail.smtp.SMTPTransp…

WSDM 2025 | 时间序列(time series)论文总结

AWSDM 2025于2025年3月10号到14号在德国汉诺威举行&#xff08;Hannover, Germany&#xff09; 本文总结了WSDM 2024有关时间序列&#xff08;time series&#xff09;的相关论文&#xff0c;如有疏漏&#xff0c;欢迎大家补充。&#xff08;没有时空数据相关的论文&#xff0…

反直觉导致卡关-迫击炮谜题

这个谜题&#xff0c;在两周目中先后卡了我至少三个小时&#xff0c;先后缓慢装填并发射迫击炮弹尝试了数百次。 一周目卡了很久&#xff0c;稀里糊涂的过了&#xff0c;想不到二周目还会卡那么久。 研究了很多播主的攻略&#xff0c;但还是一头雾水&#xff0c; 直到分析其…

庐山派K230学习日记4 PWM控制

1 本节介绍​ &#x1f4dd;本节您将学习如何通过将K230开发板的GPIO引脚复用为PWM功能并输出PWM信号&#xff1b;实现输出PWM信号及控制板载无源蜂鸣器发出声音。 &#x1f3c6;学习目标 1️⃣如何将GPIO引脚配置为PWM模式&#xff0c;通过40Pin排针中的部分引脚来输出PWM信号…

c语言的文件操作与文件缓冲区

目录 C语言文件操作函数汇总 简单介绍文件 为什么使用文件 什么是文件 文件名 二进制文件和文本文件 流和标准流 流 标准流 文件指针 文件的打开和关闭 文件的顺序读写 顺序读写函数介绍 文件的随机读写 fseek ftell rewind 文件读取结束的判定 文件缓冲区 缓…

嵌入式linux中socket控制与实现

一、概述 1、首先网络,一看到这个词,我们就会想到IP地址和端口号,那IP地址和端口各有什么作用呢? (1)IP地址如身份证一样,是标识的电脑的,一台电脑只有一个IP地址。 (2)端口提供了一种访问通道,服务器一般都是通过知名端口号来识别某个服务。例如,对于每个TCP/IP实…

Nginx:动静分离

什么是动静分离? 动静分离 是指将网站中的静态资源(如图片、样式表、脚本等)和动态内容(如 PHP、Python、Node.js 等后端生成的内容)分开部署和处理。这样做的好处是可以利用不同的服务器或缓存策略来优化不同类型的资源。 动静分离的好处 提高性能:静态资源可以直接从…

PADS Layout 差分线设计规则及其设计规则约束的详细过程步骤

一般我们的电路板有很多的差分线,有90欧姆的差分线,也有100欧姆的差分线,90欧姆的差分线主要是针对USB的差分线,特别是对于USB HUB的板子,那么我们就要设置差分线。一般我们设置差分线,一般要切换到Router里面来设置,如下所示: 那么设置差分对,一般要对原理图和Router…

计算机网络--路由表的更新

一、方法 【计算机网络习题-RIP路由表更新-哔哩哔哩】 二、举个例子 例1 例2

概述(讲讲python基本语法和第三方库)

我是北子&#xff0c;这是我自己写的python教程&#xff0c;主要是记录自己的学习成果方便自己日后复习&#xff0c; 我先学了C/C&#xff0c;所以这套教程中可能会将很多概念和C/C去对比&#xff0c;所以该教程大概不适合零基础的人。 it seems that python nowadays 只在人工…