并发场景——接口幂等性设计

什么是接口幂等性?

用数学公式表达就是。用人话说就是同一个接口,发起一次请求和多次请求,对服务端资源产生的影响是一致的

为什么需要幂等?(常见场景)

  1. 前端重复提交:用户手抖,连续点击多次提交按钮
  2. 网络超时重试(核心原因)
    1. 客户端发起请求------>服务端处理成功------>返回结果时网络断了
    2. 客户端以为服务端处理失败,触发重试
    3. 如果不做幂等,用户就被扣了两次钱
  3. MQ消息重复:消息队列为了保证消息不丢失,通常采用 At-least-once机制,这导致消费会拉取到重复消息。

解决方案

面试中不要只说一种,而是要根据操作类型(更新与新增)来分类回答。

方案一:数据库唯一索引------适用于新增

实现:

  • 建立表时,给核心字段(如order_id、user_id+activity_id)加上唯一索引
  • 业务代码执行INSERT
  • 如果抛出DuplicateKeyException(唯一键冲突异常),直接捕获,并告诉前端"已经处理成功"或"重复提交"。

优点:简单易实现

缺点:只能用于新增,不能用于更新


方案二:状态机幂等------适用于更新

场景:订单支付、订单发货、订单关闭

原理:利用状态流转的单向性

sql 复制代码
-- 只有当状态是"待支付"时,才允许更新为"已支付"
UPDATE orders 
SET status = 'PAID', update_time = NOW() 
WHERE id = 123456 AND status = 'UNPAID';

逻辑:

  • 第一次请求:找到status='UNPAID' 的记录,更新成功,返回 1。
  • 第二次请求:因为状态已经设为PAID,因此WHERE条件不满足,更行行数为0
  • 代码逻辑:如果更新行数为0,说明已经处理过了,直接返回成功

方案三:乐观锁------适用于累加/扣减

场景:库存扣减、更新余额

原理:加上版本号version

sql 复制代码
-- 1. 先查出来 version = 1
SELECT amount, version FROM account WHERE id = 1;

-- 2. 更新时带上 version
UPDATE account 
SET amount = amount - 100, version = version + 1 
WHERE id = 1 AND version = 1;

逻辑:如果并发重复请求,第二个请求带的version还是1,而数据库里的已经是2了,说明重复了,更新失败


方案四:Token令牌制------最终大招

如果上述方案都用不了(比如复杂表单提交、涉及多个微服务),就需要使用Token机制,这也是分布式解决方案

核心流程:

1、获取令牌:

  • 用户进入提交页面前,先调用后端接口getToken()
  • 后端生成一个全局的Token(如uuid),存入Redis,并返回给前端

2、提交请求:

  • 前端提交表单时,把这个Token放入Header传给后端

3、验证并删除:

  • 后端接受到这个请求,去Redis查这个Token是否存在,如果存在就删除(必须是原子操作,用Lua)
相关推荐
Victor35613 小时前
https://editor.csdn.net/md/?articleId=139321571&spm=1011.2415.3001.9698
后端
Victor35614 小时前
Hibernate(89)如何在压力测试中使用Hibernate?
后端
剩下了什么14 小时前
MySQL JSON_SET() 函数
数据库·mysql·json
山峰哥15 小时前
数据库工程与SQL调优——从索引策略到查询优化的深度实践
数据库·sql·性能优化·编辑器
较劲男子汉15 小时前
CANN Runtime零拷贝传输技术源码实战 彻底打通Host与Device的数据传输壁垒
运维·服务器·数据库·cann
灰子学技术15 小时前
go response.Body.close()导致连接异常处理
开发语言·后端·golang
java搬砖工-苤-初心不变15 小时前
MySQL 主从复制配置完全指南:从原理到实践
数据库·mysql
Gogo81616 小时前
BigInt 与 Number 的爱恨情仇,为何大佬都劝你“能用 Number 就别用 BigInt”?
后端
fuquxiaoguang16 小时前
深入浅出:使用MDC构建SpringBoot全链路请求追踪系统
java·spring boot·后端·调用链分析
毕设源码_廖学姐17 小时前
计算机毕业设计springboot招聘系统网站 基于SpringBoot的在线人才对接平台 SpringBoot驱动的智能求职与招聘服务网
spring boot·后端·课程设计