接口幂等性设计实战|防止重复提交的几种方案

接口幂等性设计实战|防止重复提交的几种方案

用户点了两次提交按钮,订单创建了两个。

这种问题经常遇到,核心就是接口没做幂等。


一、什么是幂等性?

定义: 同一个操作执行多次,结果和执行一次一样。

复制代码
幂等操作:

- 查询订单(查100次结果一样)
- 删除订单(删100次结果一样)

非幂等操作:

- 创建订单(执行100次创建100个订单)
- 扣减库存(执行100次扣100次库存)

二、常见场景

场景 原因
用户重复点击 按钮没禁用
网络超时重试 客户端自动重试
消息重复消费 MQ消息重复投递

三、幂等性方案

3.1 Token机制(推荐)

原理:

  1. 进入页面时,后端生成Token返回前端
  2. 提交时带上Token
  3. 后端验证Token,成功后删除Token
  4. 重复提交时Token已删除,拒绝请求
java 复制代码
// 1. 获取Token
@GetMapping("/token")
public String getToken() {
    String token = UUID.randomUUID().toString();
    redis.setex("submit_token:" + token, 300, "1");
    return token;
}

// 2. 提交时验证
@PostMapping("/order")
public Result createOrder(@RequestHeader("X-Submit-Token") String token) {
    String key = "submit_token:" + token;
    Long deleted = redis.del(key);
    
    if (deleted == 0) {
        return Result.fail("请勿重复提交");
    }
    
    // 创建订单
    return orderService.create(order);
}

3.2 唯一索引

sql 复制代码
-- 订单表加唯一索引
ALTER TABLE orders ADD UNIQUE INDEX uk_order_no (order_no);
java 复制代码
public void createOrder(Order order) {
    try {
        orderMapper.insert(order);
    } catch (DuplicateKeyException e) {
        log.warn("订单已存在");
    }
}

3.3 乐观锁

sql 复制代码
UPDATE orders 
SET status = 'paid', version = version + 1 
WHERE id = 1 AND version = 0;

3.4 状态机

java 复制代码
public void payOrder(Long orderId) {
    Order order = orderMapper.selectById(orderId);
    
    if (order.getStatus() != OrderStatus.PENDING) {
        return;  // 幂等处理
    }
    
    // 更新状态(带状态条件)
    int rows = orderMapper.updateStatus(orderId, OrderStatus.PAID, OrderStatus.PENDING);
    if (rows == 0) {
        return;
    }
}

3.5 分布式锁

java 复制代码
public Result createOrder(OrderDTO dto) {
    String lockKey = "lock:order:" + dto.getUserId();
    
    RLock lock = redisson.getLock(lockKey);
    if (!lock.tryLock(3, 10, TimeUnit.SECONDS)) {
        return Result.fail("请勿重复提交");
    }
    
    try {
        return doCreateOrder(dto);
    } finally {
        lock.unlock();
    }
}

四、方案对比

方案 适用场景 优点 缺点
Token机制 表单提交 通用 需要前端配合
唯一索引 有唯一字段 简单 依赖数据库
乐观锁 更新操作 简单 只适合更新
状态机 状态流转 清晰 场景有限
分布式锁 复杂业务 通用 有性能损耗

五、分布式环境测试

幂等性问题往往在分布式环境才能复现,本地单机测不出来。

如果测试环境在公司内网,可以用组网工具把本地和测试环境连起来。我用的星空组网,把本地连到测试环境的Redis、数据库,模拟真实的分布式场景:

复制代码
# 本地直连测试环境Redis
redis-cli -h 192.168.188.20 -p 6379

这样本地就能测分布式锁、Token机制这些逻辑了。


六、最佳实践

6.1 组合使用

复制代码
1. 前端:按钮防重(loading)
2. 网关:请求去重(Token/RequestId)
3. 服务:业务幂等(唯一索引/状态机)
4. 数据库:唯一约束兜底

6.2 前端防重

javascript 复制代码
async function submitOrder() {
    if (this.submitting) return;
    
    this.submitting = true;
    try {
        await api.createOrder(this.form);
    } finally {
        this.submitting = false;
    }
}

总结

幂等性设计核心:

复制代码
1. 识别幂等场景(创建、更新、删除)
2. 选择合适方案(Token、唯一索引、状态机...)
3. 多层防护(前端+后端+数据库)
4. 考虑异常情况(超时、重试)

记住:宁可多做检查,不可重复执行。

有问题评论区交流~

复制代码
相关推荐
心态还需努力呀7 小时前
CANN仓库通信库:分布式训练的梯度压缩技术
分布式·cann
Coder_Boy_10 小时前
基于SpringAI的在线考试系统-相关技术栈(分布式场景下事件机制)
java·spring boot·分布式·ddd
程序员泠零澪回家种桔子13 小时前
分布式事务核心解析与实战方案
分布式
凯子坚持 c14 小时前
CANN 生态中的分布式训练利器:深入 `collective-ops` 项目实现高效多卡协同
分布式
惊讶的猫15 小时前
rabbitmq实践小案例
分布式·rabbitmq
禁默16 小时前
打破集群通信“内存墙”:手把手教你用 CANN SHMEM 重构 AIGC 分布式算子
分布式·重构·aigc
惊讶的猫17 小时前
rabbitmq初步介绍
分布式·rabbitmq
小镇敲码人17 小时前
华为CANN框架中HCCL仓库的全面解析:分布式通信的引擎
分布式·华为
User_芊芊君子18 小时前
【分布式训练】CANN SHMEM跨设备内存通信库:构建高效多机多卡训练的关键组件
分布式·深度学习·神经网络·wpf
酷酷的崽79818 小时前
CANN 开源生态解析(四):`cann-dist-train` —— 构建高效可扩展的分布式训练引擎
分布式·开源