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

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

目标

  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. 延迟触发

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

相关推荐
我命由我123451 天前
Kotlin 开发 - lateinit 关键字
android·java·开发语言·kotlin·android studio·android-studio·android runtime
aXin_ya1 天前
微服务第八天 Sentinel 四种分布式事务模式
java·数据库·微服务
Halo_tjn1 天前
Java Set集合相关知识点
java·开发语言·算法
Linsk1 天前
Java和JavaScript的关系真是雷峰和雷峰塔的关系吗?
java·javascript·oracle
许彰午1 天前
我手写了一个 Java 内存数据库(二):B+ 树的插入与分裂
java·开发语言·面试
zhouwy1131 天前
Java 快速入门笔记:从基础语法到 Spring Boot 实战
java
大飞记Python1 天前
【2026更新】Python基础学习指南(AI版)——04数据类型
开发语言·人工智能·python
极创信息1 天前
信创产品认证怎么做?信创产品测试认证的主要流程
java·大数据·数据库·金融·软件工程
SamDeepThinking1 天前
并发量就算只有2,该上锁还得上呀
java·后端·架构
Alice-YUE1 天前
【js高频八股】防抖与节流
开发语言·前端·javascript·笔记·学习·ecmascript