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

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

目标

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

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

相关推荐
lly2024062 小时前
组合模式(Composite Pattern)
开发语言
游乐码2 小时前
c#泛型约束
开发语言·c#
Dontla2 小时前
go语言Windows安装教程(安装go安装Golang安装)(GOPATH、Go Modules)
开发语言·windows·golang
chushiyunen2 小时前
python rest请求、requests
开发语言·python
铁东博客2 小时前
Go实现周易大衍筮法三变取爻
开发语言·后端·golang
baidu_huihui2 小时前
在 CentOS 9 上安装 pip(Python 的包管理工具)
开发语言·python·pip
南 阳3 小时前
Python从入门到精通day63
开发语言·python
lbb 小魔仙3 小时前
Python_RAG知识库问答系统实战指南
开发语言·python
java1234_小锋3 小时前
Java高频面试题:Springboot的自动配置原理?
java·spring boot·面试