订单超时取消与库存回滚的完整实现(延迟任务 + 状态机)

这是一个非常典型的业务闭环:下单后未支付自动取消,并且把库存扣回去。

目标

  1. 超时未支付订单自动取消
  2. 取消时库存回滚
  3. 所有操作幂等、可重试

一、订单状态机(代码层约束)

java 复制代码
public enum OrderStatus {
    UNPAID, PAID, CANCELED
}

public final class OrderStateMachine {
    private static final Map<OrderStatus, Set<OrderStatus>> ALLOWED = Map.of(
        OrderStatus.UNPAID, Set.of(OrderStatus.PAID, OrderStatus.CANCELED),
        OrderStatus.PAID, Set.of(),        // 已支付不可取消
        OrderStatus.CANCELED, Set.of()     // 终态
    );

    public static void assertCanTransfer(OrderStatus from, OrderStatus to) {
        if (!ALLOWED.getOrDefault(from, Set.of()).contains(to)) {
            throw new IllegalStateException("状态不允许流转: " + from + " -> " + to);
        }
    }
}

二、SQL 幂等更新(核心安全点)

1) 下单(预扣库存 + UNPAID)

sql 复制代码
UPDATE stock
SET quantity = quantity - 1
WHERE sku_id = ? AND quantity > 0;

INSERT INTO orders (id, user_id, status, create_time)
VALUES (?, ?, 'UNPAID', NOW());

2) 超时取消(只允许 UNPAID -> CANCELED)

sql 复制代码
UPDATE orders
SET status = 'CANCELED', cancel_time = NOW()
WHERE id = ? AND status = 'UNPAID';

3) 库存回滚(必须幂等)

sql 复制代码
UPDATE stock
SET quantity = quantity + 1
WHERE sku_id = ?;

三、延迟任务(定时取消)

你可以用 MQ 延迟消息,也可以用定时任务。下面以 MQ 延迟为例。

java 复制代码
public void createOrder(Order order) {
    // 1) 创建订单 + 预扣库存
    createOrderAndDeductStock(order);

    // 2) 发送延迟消息(30分钟后检查是否未支付)
    rabbitTemplate.convertAndSend("order.delay.exchange", "order.delay", order.getId());
}

消费端逻辑:

java 复制代码
public void handleCancel(Long orderId) {
    Order order = orderMapper.selectById(orderId);
    if (order == null) return;

    // 状态机校验
    OrderStateMachine.assertCanTransfer(order.getStatus(), OrderStatus.CANCELED);

    // 幂等取消
    int rows = orderMapper.cancelIfUnpaid(orderId);
    if (rows == 1) {
        stockMapper.rollback(order.getSkuId());
    }
}

四、幂等保护要点

  1. 状态更新必须带上原状态条件
  2. 库存回滚只在"真正取消成功"后执行
  3. MQ 重复消息不会导致二次扣库存

最后总结

订单超时取消的核心不是"定时任务",而是:

  1. 状态机约束
  2. SQL 层幂等
  3. 延迟触发

把这三件事组合起来,才是真正稳定的业务闭环。

相关推荐
源码宝5 分钟前
MES系统源码:Java8 + SpringBoot2.7 + MySQL8 + Redis,后端源码清爽易扩展
java·后端·源码·springboot·mes系统·源码二开·mes源码
JAVA社区25 分钟前
Java高级全套教程(十)—— SpringCloudAlibaba超详细实战详解
java·开发语言·spring cloud·面试·职场和发展
弥树子32 分钟前
踩坑记录:服务器内网调用接口,真实请求URL与官方公开URL不一致问题排查
开发语言·php
金銀銅鐵33 分钟前
[Java] 如何理解 class 文件中方法的 descriptor?
java·后端
云烟成雨TD1 小时前
Spring AI Alibaba 1.x 系列【63】AI Agent 长期记忆
java·人工智能·spring
z落落1 小时前
C# ToCharArray + foreach遍历 + String与StringBuilder
开发语言·c#
憧憬成为java架构高手的小白1 小时前
苍穹外卖--day09
java·spring boot·百度
学代码的真由酱1 小时前
Java多用户一对一网页聊天室-测试报告
java·开发语言·功能测试·测试
人道领域1 小时前
【LeetCode刷题日记】669.修剪二叉搜索树
开发语言·python·算法
Jasonakeke2 小时前
SpringBoot自动配置原理揭秘
java·spring boot·后端