优惠券秒杀的核心业务逻辑

这是一套 SpringBoot + MyBatis-Plus 实现的优惠券秒杀下单功能,前端用户点击抢购后,后端校验时间、库存,然后扣库存、生成订单并返回结果。

这是一个典型的 Java Spring Boot 后端业务代码 ,实现了优惠券秒杀(Seckill)的核心业务逻辑。我们可以将这段代码拆解为 3 个核心层面来理解:

(1)校验:检查秒杀是否开始、是否结束,以及库存是否足够。

(2)扣减:如果校验通过,减少数据库中的优惠券库存。

(3)创建:在订单表中生成一条订单记录。

(4)返回:告诉用户是否成功,并返回订单 ID。

Controller

复制代码
@RestController
@RequestMapping("/voucher-order")
public class VoucherOrderController {

    @Resource
    private IVoucherOrderService voucherOrderService;

    @PostMapping("seckill/{id}")
    public Result seckillVoucher(@PathVariable("id") Long voucherId) {
        return voucherOrderService.seckillVoucher(voucherId);
    }
}

Service

复制代码
public interface IVoucherOrderService extends IService<VoucherOrder> {
    Result seckillVoucher(Long voucherId);
}

ServiceImpl

复制代码
@Service
public class VoucherOrderServiceImpl extends ServiceImpl<VoucherOrderMapper, VoucherOrder> implements IVoucherOrderService {

    @Resource
    private ISeckillVoucherService seckillVoucherService;

    @Resource
    private RedisIdWorker redisIdWorker;

    @Override
    @Transactional
    public Result seckillVoucher(Long voucherId) {
        // 1.查询优惠券信息
        SeckillVoucher voucher = seckillVoucherService.getById(voucherId);

        // 2.判断秒杀是否开始
        if (voucher.getBeginTime().isAfter(LocalDateTime.now())) {
            return Result.fail("秒杀尚未开始!");
        }

        // 3.判断秒杀是否结束
        if (voucher.getEndTime().isBefore(LocalDateTime.now())) {
            return Result.fail("秒杀已经结束!");
        }

        // 4.判断库存是否充足
        if (voucher.getStock() < 1) {
            return Result.fail("库存不足!");
        }

        // 5.扣减库存
        boolean success = seckillVoucherService.update()
                .setSql("stock = stock - 1")
                .eq("voucher_id", voucherId)
                .update();
        if (!success) {
            return Result.fail("库存不足!");
        }

        // 6.创建订单
        VoucherOrder voucherOrder = new VoucherOrder();
        // 订单ID
        long orderId = redisIdWorker.nextId("order");
        voucherOrder.setId(orderId);
        // 用户ID
        Long userId = UserHolder.getUser().getId();
        voucherOrder.setUserId(userId);
        // 代金券ID
        voucherOrder.setVoucherId(voucherId);
        save(voucherOrder);

        // 7.返回订单号
        return Result.ok(orderId);
    }
}

整体结构(三层)

  1. Controller 层

    • 接收前端 HTTP 请求
    • 只做转发,不写业务逻辑
    • 接口地址:/voucher-order/seckill/{优惠券ID}
  2. Service 接口

    • 定义秒杀方法
    • 方便解耦、扩展、测试
  3. Service 实现类(核心)

    • 所有业务逻辑都在这里
    • 加了事务 @Transactional,保证扣库存和生成订单要么都成功,要么都失败

整体执行流程(完整逻辑)

  1. 前端传入优惠券 ID,请求秒杀接口
  2. 后端根据 ID 查询优惠券信息
  3. 校验秒杀时间:
    • 没开始 → 拒绝
    • 已结束 → 拒绝
  4. 校验库存:
    • 库存不足 → 拒绝
  5. 库存充足 → 执行数据库扣减库存
  6. 扣减成功 → 生成全局唯一订单 ID
  7. 创建订单记录(包含用户 ID、优惠券 ID)
  8. 保存订单到数据库
  9. 返回订单号给前端,表示秒杀成功

涉及的数据库表

1.tb_seckill_voucher:秒杀优惠券表,存库存、开始结束时间

2.tb_voucher_order:用户秒杀成功后的订单表


关键点总结

1.这是标准后端秒杀业务

2.用了 Spring 事务 保证数据安全

3.用了 Redis 生成唯一订单号

4.用 MyBatis-Plus 简化数据库操作

5.流程:校验时间 → 校验库存 → 扣库存 → 生成订单

相关推荐
前端之虎陈随易6 分钟前
2年没用Nodejs了,Bun很香
linux·前端·javascript·vue.js·typescript
苍煜10 分钟前
SpringBoot AOP切面编程精讲:实现方式、Spring区别及与自定义注解生产实战
java·spring boot·spring
yzs8713 分钟前
从Hydra到storage_engine:PostgreSQL列存引擎的性能跃迁与技术进化
数据库·postgresql
胡小禾14 分钟前
企业内部文件处理方案
java
红云梦16 分钟前
官方 Anthropic Postgres MCP Server 存在 SQL 注入漏洞 -- SafeDB 是如何做到 4 层防御的
数据库·sql
TDengine (老段)24 分钟前
红有软件重构智能油田时序数据底座,支撑生产实时感知与设备预测性维护
大数据·数据库·人工智能·重构·时序数据库·tdengine
Hooray26 分钟前
用时7天,花费30元,我vibe coding这个网站
前端·agent·ai编程
小小高不懂写代码40 分钟前
RAG--检索增强生成--原理及实战
前端·人工智能
倒霉蛋小马41 分钟前
【Redis】什么是缓存击穿?
数据库·redis·缓存
空中海42 分钟前
04 工程化、质量体系与 React 生态
前端·ubuntu·react.js