当前位置: 首页 > news >正文

二刷 黑马点评 秒杀优化

优化逻辑

把耗时较短的逻辑判断放入redsi中,比如库存是否足够以及是否一人一单,只要这样的逻辑完成,就代表一定能下单成功,我们就将结果返回给用户,然后我们再开一个线程慢慢执行队列中的信息

问题:
如何快速校验一人一单以及库存是否充足

交验和下单是两个线程,如何将二者对应:
在redis操作完成之后,会返回一些信息给前端,同时将这些信息丢给异步队列执行,后续操作通过id来查询下单逻辑是否完成

![[Pasted image 20250717173831.png]]

整体流程

下单后判断是否充足只需要去redis根据key查询对应的value是否大于0 ,如果大于0再判断是否下过单,如果在set集合中没有这条数据,那么就将userId和优惠卷存入redis,将优惠卷id、用户id和订单id存入阻塞队列中,异步存储到数据库

![[Pasted image 20250717174050.png]]

stringRedisTemplate.opsForValue().set(SECKILL_STOCK_KEY + voucher.getId(), voucher.getStock().toString());
保存优惠卷并将保存秒杀的库存到Redis

-- 3.脚本业务
-- 3.1.判断库存是否充足 get stockKey
if(tonumber(redis.call('get', stockKey)) <= 0) then-- 3.2.库存不足,返回1return 1
end
-- 3.2.判断用户是否下单 SISMEMBER orderKey userId
if(redis.call('sismember', orderKey, userId) == 1) then-- 3.3.存在,说明是重复下单,返回2return 2
end
-- 3.6.发送消息到队列中, XADD stream.orders * k1 v1 k2 v2 ...
redis.call('xadd', 'stream.orders', '*', 'userId', userId, 'voucherId', voucherId, 'id', orderId)

只要>0就可以下单,然后判断用户是否下过单

命令结构
XADD stream.orders * k1 v1 k2 v2 ...
- stream.orders:目标流的名称。
- *:自动生成唯一的消息 ID(格式为 时间戳-序列号)。
- k1 v1 k2 v2 ...:消息的键值对数据。

Long result = stringRedisTemplate.execute(  SECKILL_SCRIPT, Collections.emptyList(),  voucherId.toString(), userId.toString(), String.valueOf(orderId)  
);

脚本参数
- SECKILL_SCRIPT:预定义的 Lua 脚本,处理秒杀业务逻辑(如库存校验、扣减)。
- Collections.emptyList():Lua 脚本中KEYS参数为空列表(无需键名参数)。
- 后续参数为ARGV数组,依次是voucherIduserIdorderId,供 Lua 脚本内部使用。

private static final ExecutorService SECKILL_ORDER_EXECUTOR=Executors.newSingleThreadExecutor();
定义了一个静态常量线程池,这是一个单线程的执行器,保证任务按顺序执行, 避免多线程并发处理同一用户订单导致的重复下单问题

// 类初始化后启动工作线程 
@PostConstruct 
private void init() { SECKILL_ORDER_EXECUTOR.submit(new VoucherOrderHandler()); }

@PostConstruct确保类初始化后立即启动订单处理线程,并会持续运行直到应用关闭

 private class VoucherOrderHandler implements Runnable{@Overridepublic void run() {while (true){try {// 1.获取队列中的订单信息VoucherOrder voucherOrder = orderTasks.take();// 2.创建订单handleVoucherOrder(voucherOrder);} catch (Exception e) {log.error("处理订单异常", e);}}}}

实现 Runnable 接口的类通常用于创建线程任务,可以通过 Thread 类或线程池执行。

  • orderTasks.take() 是阻塞调用,队列空时线程会等待
  • 确保订单按放入队列的顺序处理
  • 异常处理保证线程不会因异常终止
    proxy.createVoucherOrder(voucherOrder);
  • Spring 事务依赖 ThreadLocal,多线程环境下子线程无法获取主线程的事务上下文
  • 通过注入代理对象proxy调用事务方法,确保事务生效
    也就是说继承Runnable接口的类可以被在线程池中被调用
    而这里选择了在类初始化之后就调用
Spring 事务管理的工作原理
private void handleVoucherOrder(VoucherOrder voucherOrder) {// ...获取锁...try {// 通过代理对象调用事务方法proxy.createVoucherOrder(voucherOrder);} finally {// ...释放锁...}
}

Spring 的声明式事务是通过 AOP 代理实现的。当在方法上使用@Transactional注解时,Spring 会为该类创建一个代理对象,在调用带注解的方法时,代理会拦截调用并添加事务管理逻辑。
如果直接使用this.createVoucherOrder(voucherOrder),事务不会生效,因为AOP代理被绕过了,必须通过代理对象调用这个方法才能触发事务增强
因为:

  • Spring 事务是基于ThreadLocal实现的,不同线程有独立的ThreadLocal副本
  • 子线程(如线程池中的工作线程)无法获取主线程的事务上下文
  • 必须通过代理对象调用才能确保事务拦截器被触发

使用AopContext.currentProxy():在方法内部获取当前代理对象

总结

1. 为什么将库存校验和一人一单判断放在 Redis 中执行?

将高频、低耗时的校验逻辑放在 Redis 中执行,利用其内存级别的读写性能和原子性操作能力,可以快速完成资格校验。同时避免了直接访问数据库带来的网络延迟和 IO 开销,显著提升系统吞吐量。

2. 如何保证库存扣减和订单记录的原子性?

通过 Lua 脚本在 Redis 端实现原子操作。脚本中先校验库存和用户下单状态,若满足条件则直接扣减库存并记录订单信息,整个过程不可分割,有效防止超卖和重复下单问题。

3. 异步处理订单时,如何保证数据最终一致性?

采用消息队列实现异步解耦,主流程完成 Redis 操作后立即返回结果,同时将订单信息发送到阻塞队列。独立线程按顺序处理队列中的订单,确保数据最终一致性。即使处理过程中出现异常,也可通过重试机制保证订单最终入库。

4. 为什么使用单线程执行器处理订单队列?

使用单线程执行器(Executors.newSingleThreadExecutor())可以确保同一用户的订单按顺序处理,避免多线程并发处理导致的重复下单问题。同时保证了操作的顺序性,与 Redis 中的校验逻辑形成完整闭环。

5. 在多线程环境下,如何保证 Spring 事务生效?

在子线程中通过注入代理对象调用事务方法,而非直接使用this引用。因为 Spring 事务基于 AOP 代理和ThreadLocal实现,子线程无法直接获取主线程的事务上下文。通过代理对象调用可确保事务拦截器被触发,从而正确管理事务。

6. Redis 消息队列相比传统阻塞队列有什么优势?

Redis 的 Stream 数据结构支持持久化和多消费者组,相比 Java 内置的阻塞队列,具有更好的可靠性和扩展性。即使服务重启,未处理的消息也不会丢失,适合分布式系统下的异步通信场景。

http://www.dtcms.com/a/284238.html

相关文章:

  • 板凳-------Mysql cookbook学习 (十二--------1)
  • 医院各类不良事件上报,PHP+vscode+vue2+element+laravel8+mysql5.7不良事件管理系统源代码,成品源码,不良事件管理系统
  • React事件处理
  • 【FFmpeg 快速入门】本地播放器 项目
  • c++:explicit关键字
  • Python枚举技巧:轻松获取索引与值
  • 【Linux手册】缓冲区:深入浅出,从核心概念到实现逻辑
  • Python爬虫入门到实战(2)-selenium驱动浏览器
  • 8.预处理-demo
  • 建筑兔零基础人工智能自学记录111|初识comfyui-20
  • PyTorch笔记8----------卷积神经网络
  • 使用Pytorch进行数字手写体识别
  • 对比分析:给数据找个 “参照物”,让孤立数字变 “决策依据”
  • notepad++ 多行复制拼接
  • 原生前端JavaScript/CSS与现代框架(Vue、React)的联系与区别(详细版)
  • Git 子模块只更新部分模块的问题排查总结
  • Elasticsearch+Logstash+Filebeat+Kibana部署【7.1.1版本】
  • GitHub Jekyll博客本地Win开发环境搭建
  • 【URL 转换为PDF】HTML转换为PDF
  • 【哈希映射实现的并集查找】P5962 [BalticOI 2004] ships 船|普及+
  • 【析精】Landmark-Guided Subgoal Generation in Hierarchical Reinforcement Learning
  • 【加解密与C】Base系列(六)Base100
  • 基于在线地图的路径规划测评对比-综合对比城区、农村及城乡结合处的导航
  • JavaScript进阶篇——第八章 原型链、深浅拷贝与原型继承全解析
  • 20250717 Ubuntu 挂载远程 Windows 服务器上的硬盘
  • Linux C 进程基本操作
  • 冒泡排序、选择排序、插入排序、快速排序
  • NLP——迁移学习
  • 【unity组件介绍】URP Decal Projector贴花投影器,将特定材质(贴花)投影到场景中的其他对象上。
  • RabbitMQ深度解析:从核心概念到实战应用