接口重试的7种常用方案!
前言
记得五年前的一个深夜,某个电商平台的订单退款接口突发异常,因为银行系统网络抖动,退款请求连续失败。
原本技术团队只是想“好心重试几次”,结果开发小哥写的重试代码竟疯狂调用了银行的退款接口 82次!
最终导致用户账户重复退款,平台损失过百万。
老板在复盘会上质问:“接口重试这么基础的事,为什么还能捅出大篓子?”
大家哑口无言,因为所有人都以为只要加个 for
循环,再睡几秒就完事了……
这篇文章跟大家一起聊聊重试的7种常用方案,希望对你会有所帮助。
1 暴力轮回法
问题场景
某实习生写的用户注册短信发送接口。
在一个while循环中,重复调用第三方的发短信接口给用户发送短信。
代码如下:
public void sendSms(String phone) {int retry = 0;while (retry < 5) { // 无脑循环try {smsClient.send(phone);break;} catch (Exception e) {retry++;Thread.sleep(1000); // 固定1秒睡眠}}
}
事故现场
某次短信服务器出现了过载问题,导致所有请求都延迟了3秒。
这个暴力循环的代码在 0.5秒内同时发起数万次重试,直接打爆短信平台,触发了 熔断封禁,连正常请求也被拒绝。
教训
-
💥 不做延迟间隔调整:固定间隔导致重试请求集中爆发
-
💥 无视异常类型:非临时性错误(如参数错误)也尝试重试
-
🔑 修复方案:加上随机的重试间隔,并过滤不可重试的异常
2 Spring Retry
应用场景
Spring Retry适用于中小项目,通过注解快速实现基本重试和熔断(如订单状态查询接口)。
通过声明@Retryable注解,来实现接口重试的功能。
配置示例
@Retryable(value = {TimeoutException.class}, // 只重试超时异常maxAttempts = 3,backoff = @Backoff(delay = 1000, multiplier = 2) // 1秒→2秒→4秒
)
public boolean queryOrderStatus(String orderId) {return httpClient.get("/order/" + orderId);
}@Recover // 兜底回退方法
public boolean fallback() {return false;
}
优势
-
声明式注解:代码简洁,与业务逻辑解耦
-
指数退避:自动拉长重试间隔
-
熔断集成:结合
@CircuitBreaker
可快速阻断异常流量
3 Resilience4j
高阶场景
对于有些需要自定义退避算法、熔断策略和多层防护的大中型系统(如支付核心接口),我们可以使用 Resilience4j。
核心代码如下:
// 1. 重试配置:指数退避 + 随机抖动
RetryConfig retryConfig = RetryConfig.custom().maxAttempts(3).intervalFunction(IntervalFunction.ofExponentialRandomBackoff(1000L, // 初始间隔1秒2.0, // 指数倍数0.3 // 随机抖动系数)).retryOnException(e -> e instanceof TimeoutException).build();// 2. 熔断配置:错误率超50%时熔断
CircuitBreakerConfig cbConfig = CircuitBreakerConfig.custom().slidingWindow(10, 10, CircuitBreakerConfig.SlidingWindowType.COUNT_BASED) .failureRateThreshold(50).build();// 组合使用
Retry retry = Retry.of("payment", retryConfig);
CircuitBreaker cb = CircuitBreaker.of("payment", cbConfig);// 执行业务逻辑
Supplier<Boolean> supplier = () -> paymentService.pay();
Supplier<Boolean> decorated = Decorators.ofSupplier(supplier).withRetry(retry).withCircuitBreaker(cb).decorate();
效果
某电商大厂上线此方案后,支付接口 超时率下降60% ,且熔断触发频率降低近 90%
真正做到了“打不还手,骂不还口”。
4 MQ队列
适用场景
高并发、允许延时的异步场景(如物流状态同步)。
实现原理
-
首次请求失败后,将消息投递至 延时队列
-
队列根据预设的延时时间(如5秒、30秒、1分钟)重试消费
-
若达到最大重试次数,则转存至 死信队列(人工处理)
RocketMQ代码片段如下:
// 生产者发送延时消息
Message<String> message = new Message();
message.setBody("订单数据");
message.setDelayTimeLevel(3); // RocketMQ预设的10秒延迟级别
rocketMQTemplate.send(message);// 消费者重试
@RocketMQMessageListener(topic = "DELAY_TOPIC")
public class DelayConsumer {@Overridepublic void handleMessage(Message message) {try {syncLogistics(message);} catch (Exception e) {// 重试次数 + 1,并重新发送到更高延迟级别resendWithDelay(message, retryCount + 1);}}
}
如何RocketMQ的消费者消费失败,会自动发起重试。
5 定时任务
适用场景
对于有些不需要实时反馈,允许批量处理的任务(如文件导入)的业务场景,我们可以使用定时任务。
在这里以Quartz为例。
具体代码如下:
@Scheduled(cron = "0 0/5 * * * ?") // 每5分钟执行
public void retryFailedTasks() {List<FailedTask> list = failedTaskDao.listUnprocessed(5); // 查失败任务list.forEach(task -> {try {retryTask(task);task.markSuccess();} catch (Exception e) {task.incrRetryCount();}failedTaskDao.update(task);});
}
6 两阶段提交
适用场景
对于严格保证数据一致性的场景(如资金转账),我们可以使用两阶段提交机制。
关键实现
-
第一阶段:记录操作流水到数据库(状态为“进行中”)
-
第二阶段:调用远程接口,并根据结果更新流水状态
-
定时补偿:扫描超时的“进行中”流水重新提交
大致代码如下:
@Transactional
public void transfer(TransferRequest req) {// 1. 记录流水transferRecordDao.create(req, PENDING);// 2. 调用银行接口boolean success = bankClient.transfer(req);// 3. 更新流水状态transferRecordDao.updateStatus(req.getId(), success ? SUCCESS : FAILED);// 4. 失败转异步重试if (!success) {mqTemplate.send("TRANSFER_RETRY_QUEUE", req);}
}
7 分布式锁
应用场景
对于一些多服务实例、多线程环境的防重复提交(如秒杀)的业务场景,我们可以使用分布式锁。
这里以Redis + Lua的分布式锁为例。
代码如下:
public boolean retryWithLock(String key, int maxRetry) {String lockKey = "api_retry_lock:" + key;for (int i = 0; i < maxRetry; i++) {// 尝试获取分布式锁if (redis.setnx(lockKey, "1", 30, TimeUnit.SECONDS)) {try {return callApi();} finally {redis.delete(lockKey);}}Thread.sleep(1000 * (i + 1)); // 等待释放锁}return false;
}
总结
重试就像机房里的灭火器——永远不希望用到它,但必须保证关键时刻能救命。
我们工作中选择哪种方案?
别只看技术潮流,而要看业务的长矛和盾牌,需要哪种配合。
最后送大家一句话:系统稳定的秘诀,是永远对重试保持敬畏。
文章转载自:苏三说技术
原文链接:接口重试的7种常用方案! - 苏三说技术 - 博客园
体验地址:JNPF快速开发平台
相关文章:
接口重试的7种常用方案!
前言 记得五年前的一个深夜,某个电商平台的订单退款接口突发异常,因为银行系统网络抖动,退款请求连续失败。 原本技术团队只是想“好心重试几次”,结果开发小哥写的重试代码竟疯狂调用了银行的退款接口 82次! 最终导致…...

vue3:Table组件动态的字段(列)权限、显示隐藏和左侧固定
效果展示 根据后端接口返回,当前登录用户详情中的页面中el-table组件的显示隐藏等功能。根据菜单id查询该菜单下能后显示的列。 后端返回的数据类型: 接收到后端返回的数据后处理数据结构. Table组件文件 <!-- 自己封装的Table组件文件 --> onMounted(()>…...

pikachu靶场通关笔记13 XSS关卡09-XSS之href输出
目录 一、href 1、常见取值类型 2、使用示例 3、安全风险 二、源码分析 1、进入靶场 2、代码审计 3、渗透思路 三、渗透实战 1、注入payload1 2、注入payload2 3、注入payload3 本系列为通过《pikachu靶场通关笔记》的XSS关卡(共10关)渗透集合ÿ…...

MCP客户端Client开发流程
1. uv工具入门使用指南 1.1 uv入门介绍 MCP开发要求借助uv进行虚拟环境创建和依赖管理。 uv 是一个Python 依赖管理工具,类似于pip 和 conda ,但它更快、更高效,并且可以更好地管理 Python 虚拟环境和依赖项。它的核心目标是 替代 pip 、…...

学习日记-day21-6.3
完成目标: 目录 知识点: 1.集合_哈希表存储过程说明 2.集合_哈希表源码查看 3.集合_哈希表无索引&哈希表有序无序详解 4.集合_TreeSet和TreeMap 5.集合_Hashtable和Vector&Vector源码分析 6.集合_Properties属性集 7.集合_集合嵌套 8.…...

C语言探索之旅:深入理解结构体的奥秘
目录 引言 一、什么是结构体? 二、结构体类型的声明和初始化 1、结构体的声明 2、结构体的初始化 3、结构体的特殊声明 4、结构体的自引用 5、结构体的重命名 三、结构体的内存对齐 1、对齐规则 2、为什么存在内存对齐? 3、修改默认对齐数 三…...
uniapp 开发企业微信小程序,如何区别生产环境和测试环境?来处理不同的服务请求
在 uniapp 开发企业微信小程序时,区分生产环境和测试环境是常见需求。以下是几种可靠的方法,帮助你根据环境处理不同的服务请求: 一、通过条件编译区分(推荐) 使用 uniapp 的 条件编译 语法,在代码中标记…...
Dockerfile常用指令介绍
Dockerfile常用指令介绍 Dockerfile是一个文本文件,用于定义Docker镜像的构建过程。下面介绍一些最常用的Dockerfile指令及其用法: 基础指令 FROM - 指定基础镜像 FROM python:3.9-slim这是Dockerfile的第一个指令,用于指定构建镜像的基础镜…...
Docker 容器化:核心技术原理与实践
哈喽,大家好,我是左手python! Docker 的基本概念与核心组件 Docker 是一个开源的容器化平台,能够将应用程序及其依赖项打包成一个容器,确保在任何环境中都能一致运行。Docker 的核心在于其容器化技术,这种…...
不确定性分析在LEAP能源-环境系统建模中的整合与应用
本内容突出与实例结合,紧密结合国家能源统计制度及《省级温室气体排放编制指南》,深入浅出地介绍针对不同级别研究对象时如何根据数据结构、可获取性、研究目的,构建合适的能源生产、转换、消费、温室气体排放(以碳排放为主&#…...

经典算法回顾之最小生成树
最小生成树(Minimum Spanning Tree,简称MST)是图论中的一个重要概念,主要用于解决加权无向图中连接所有顶点且总权重最小的树结构问题。本文对两种经典的算法即Prim算法和Kruskal算法进行回顾,并对后者的正确性给出简单…...

Ubuntu下实现nginx反向代理
1. 多个ngx实例安装 脚本已经在deepseek的指导下完成啦! deepseek写的脚本支持ubuntu/centos两种系统。 ins_prefix"/usr/local/" makefile_gen() {ngx$1 ngx_log_dir"/var/log/"$ngx"/"ngx_temp_path"/var/temp/"${ngx}…...

c++ QicsTable使用实例
效果图: #include <QicsTable.h> #include <QicsDataModelDefault.h> #include <QVBoxLayout> Demo1::Demo1(QWidget *parent) : QWidget(parent) { ui.setupUi(this); const int numRows 10; const int numCols 5; // create th…...

在WordPress上添加隐私政策页面
在如今的互联网时代,保护用户隐私已经成为每个网站管理员的责任。隐私政策不仅是法律要求,还能提高用户对网站的信任。本文将介绍两种常用方法,帮助你在WordPress上轻松创建并发布隐私政策页面。这些方法简单易行,符合中国用户的阅…...
二维 根据矩阵变换计算镜像旋转角度
在二维变换中,镜像(Reflection) 是一种特殊的线性变换,它会将图形对称地翻转到某个轴线或点。镜像的存在会显著影响圆弧变换后的参数(圆心、半径、起始角度),尤其是在角度方向和旋转方向的处理上…...
你工作中涉及的安全方面的测试有哪些怎么回答
在面试或工作总结中,回答 **“工作中涉及的安全测试”** 时,可以结合具体场景、测试方法和工具,突出你的技术广度和深度。以下是结构化回答建议: --- ### **1. 分类说明安全测试范围** #### **(1) Web 应用安全测试** - **OWASP…...

阿里云ACP云计算备考笔记 (3)——云服务器ECS
目录 第一章 整体概览 第二章 ECS简介 1、产品概念 2、ECS对比本地IDC 3、BGP机房优势 第三章 ECS实例 1、实例规格族 2、实例系列 3、应用场景推荐选型 4、实例状态 5、创建实例 ① 完成基础配置 ② 完成网络和安全组配置 ③ 完成管理配置和高级选项 ④ 确认下单…...
Eigen实现非线性最小二乘拟合 + Gauss-Newton算法
下面是使用 Eigen 实现的 非线性最小二乘拟合 Gauss-Newton 算法 的完整示例,拟合模型为: 拟合目标模型: y exp ( a x 2 b x c ) y \exp(a x^2 b x c) yexp(ax2bxc) 已知一组带噪声数据点 ( x i , y i ) (x_i, y_i) (xi,yi)&…...
区块链技术:原理、应用与发展趋势
区块链技术:原理、应用与发展趋势 引言 区块链作为一种去中心化的分布式账本技术,自2008年比特币白皮书发布以来,已经从简单的加密货币底层技术发展成为具有广泛应用前景的创新技术。区块链通过独特的数据结构和加密机制,实现了…...

从零开始:用Tkinter打造你的第一个Python桌面应用
目录 一、界面搭建:像搭积木一样组合控件 二、菜单系统:给应用装上“控制中枢” 三、事件驱动:让界面“活”起来 四、进阶技巧:打造专业级体验 五、部署发布:让作品触手可及 六、学习路径建议 在Python生态中&am…...

Web开发主流前后端框架总结
🖥 一、前端主流框架 前端框架的核心是提升用户界面开发效率,实现高交互性应用。当前三大主流框架各有侧重: React (Meta/Facebook) 核心特点:采用组件化架构与虚拟DOM技术(减少真实DOM操作,优化渲染性能&…...
Java Spring Boot 自定义注解详解与实践
目录 一、自定义注解的场景与优势1.1 场景1.2 优势 二、创建自定义注解2.1 定义注解2.2 创建注解处理器 三、使用自定义注解3.1 在业务方法上使用注解3.2 配置类加载注解 四、总结 在 Spring Boot 中,自定义注解为我们提供了一种灵活且强大的方式来简化开发、增强代…...

GlobalSign、DigiCert、Sectigo三种SSL安全证书有什么区别?
GlobalSign、DigiCert和Sectigo是三家知名的SSL证书颁发机构,其产品在安全性、功能、价格和适用场景上存在一定差异。选择SSL证书就像为你的网站挑选最合身的“安全盔甲”,核心是匹配你的实际需求,避免过度配置或防护不足。 一、核心特点对…...

力扣面试150题--二叉搜索树中第k小的元素
Day 58 题目描述 思路 直接采取中序遍历,不过我们将k参与到中序遍历中,遍历到第k个元素就结束 /*** Definition for a binary tree node.* public class TreeNode {* int val;* TreeNode left;* TreeNode right;* TreeNode() {}* …...

SQL Server Agent 不可用怎么办?
在 SQL Server Management Studio (SSMS) 中,SQL Server Agent 通常位于对象资源管理器(Object Explorer)的树形结构中,作为 SQL Server 实例的子节点。以下是详细说明和可能的原因: 1. SQL Server Agent 的位置 默认路…...

css-塞贝尔曲线
文章目录 1、定义2、使用和解释 1、定义 cubic-bezier() 函数定义了一个贝塞尔曲线(Cubic Bezier)语法:cubic-bezier(x1,y1,x2,y2) 2、使用和解释 x1,y1,x2,y2,表示两个点的坐标P1(x1,y1),P2(x2,y2)将以一条直线放在范围只有 1 的坐标轴中,并…...
Java并发编程哲学系列汇总
文章目录 并发编程基础并发编程进阶并发编程实践 并发编程基础 Java并发编程基础小结 Java线程池知识点小结 详解JUC包下各种锁的使用 并发编程利器Java CAS原子类全解 深入理解Java中的final关键字 Java并发容器深入解析:HashMap与ArrayList线程安全问题及解…...

docker使用proxy拉取镜像
前提条件,宿主机可以访问docker hub 虚拟机上telnet 宿主机7890能正常访问 下面的才是关键,上面部分自己想办法~ 3. 编辑 /etc/docker/daemon.json {"proxies": {"http-proxy": "http://192.168.100.1:7890","ht…...

服务端定时器的学习(一)
一、定时器 1、定时器是什么? 定时器不仅存在于硬件领域,在软件层面(客户端、网页和服务端)也普遍应用,核心功能都是高效管理大量延时任务。不同应用场景下,其实现方式和使用方法有所差异。 2、定时器解…...
【前端】vue 防抖和节流
在 Vue.js 中,防抖(Debounce) 和 节流(Throttle) 是优化高频事件(如输入、滚动、点击)的核心技术,可显著提升性能与用户体验。以下是具体实现方法和最佳实践: ⏳ 一、防抖…...