优化逻辑

把耗时较短的逻辑判断放入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.pswp.cn/web/89298.shtml
繁体地址,请注明出处:http://hk.pswp.cn/web/89298.shtml
英文地址,请注明出处:http://en.pswp.cn/web/89298.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

HANA SQLScript中的变量类型汇总

在 SAP HANA SQLScript 中&#xff0c;可以使用多种变量类型&#xff0c;包括标量&#xff08;Scalar&#xff09;类型、表类型和结构化类型。以下是各种变量类型的详细说明和示例。1. 标量变量&#xff08;Scalar Variables&#xff09; 标量变量是用于存储单个值&#xff08;…

基于 Amazon Nova Sonic 和 MCP 构建语音交互 Agent

1、引言 随着人工智能技术的飞速发展&#xff0c;自然语言处理和语音交互技术正在深刻改变人机交互的方式。语音交互正从简单的“机械应答”向更自然的“类人对话”演进 。传统的语音系统通常采用模块化架构&#xff0c;将语音处理流程割裂为 ASR&#xff08;自动语音识别&…

项目的存量接口怎么低成本接入MCP?

项目的存量接口怎么低成本接入MCP&#xff1f; 老项目里的一些接口&#xff0c;如何低成本的接入MCP&#xff08;0成本不可能&#xff09;&#xff0c;变成MCP server 的tools&#xff1f; 先抛出这个问题&#xff1f;评论区的xdm如果有懂的&#xff0c;可以打在评论区&#xf…

用图片生成高保真3D模型!Hi3DGen以法线为桥,为高清三维几何生成另辟蹊径

主页&#xff1a;http://qingkeai.online/ 原文&#xff1a;用图片生成高保真3D模型&#xff01;Hi3DGen以法线为桥&#xff0c;为高清三维几何生成另辟蹊径 随着从二维图像构建高保真三维模型的需求日益增长&#xff0c;现有方法由于域间隙的限制以及 RGB 图像固有的模糊性&a…

Charles抓包工具中文安装和使用详解,快速掌握API调试与网络优化

Charles抓包工具中文安装和使用详解 在软件开发中&#xff0c;调试API请求、捕获网络流量以及优化应用性能是开发者日常工作中不可或缺的环节。Charles抓包工具作为业内领先的网络调试工具&#xff0c;以其功能强大、易用性高、支持HTTPS流量解密等特点&#xff0c;广泛应用于A…

Java :List,LinkedList,ArrayList

文章目录List常用方法List集合的遍历方式ArrayList底层的原理LinkedList底层原理常用方法List常用方法 //1.创建一个ArrayList集合对象&#xff08;有序、有索引、可以重复&#xff09; List<String> list new ArrayList<>(); list.add("蜘蛛精"); list…

LLM面试题及讲解 4

LLM面试题及讲解 4 目录 LLM面试题及讲解 4 题目讲解 一、基础概念与理论 二、模型训练与优化 三、应用与实践 四、前沿研究与趋势 大型语言模型(LLM)的核心特征是什么? LLM与传统NLP技术的本质区别是什么? Transformer架构的基本组成部分有哪些?其在LLM中为何重要? BERT…

Harmony-Next鸿蒙实战开发项目-仿小米商城App----V2

1.、简介 本项目是Harmony-Next原生开发&#xff0c;真实网络请求。采用V2等状态管理装饰器。包含&#xff08;首页、分类、发现、购物车、我的、登录、搜索&#xff0c;搜索结果&#xff0c;商品详情等&#xff09;.包含V2对接口返回数据的深度监听。 2、页面展示&#xff1…

python闭包和装饰器(超详解)

目录 一、闭包的概念 1.概念 2.闭包的特征 3.闭包的作用 二、装饰器 1.什么是装饰器 2.装饰器的作用 1.统计代码耗时 2.对代码进行权限检查 3.记录日志 3.闭包和装饰器的关系 4.注意事项&#xff1a; 一、闭包的概念 1.概念 闭包&#xff08;Closure&#xff09;指…

解决hadoop常用到的问题

1.namenode无法启动问题 报错1. ERROR: Attempting to operate on hdfs namenode as root ERROR: but there is no HDFS_NAMENODE_USER defined. 原因&#xff1a;不能用 root 用户直接启动 Hadoop 的 HDFS 组件&#xff08;NameNode / DataNode / SecondaryNameNode&#xff0…

深度学习G3周:CGAN入门(生成手势图像)

&#x1f368; 本文为&#x1f517;365天深度学习训练营中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊 基础任务&#xff1a; 1.条件生成对抗网络&#xff08;CGAN&#xff09;的基本原理 2.CGAN是如何实现条件控制的 3.学习本文CGAN代码&#xff0c;并跑通代码…

流式数据处理实战:用状态机 + scan 优雅过滤 AI 响应中的 `<think>` 标签

流式数据处理实战&#xff1a;用状态机 scan 优雅过滤 AI 响应中的 <think> 标签 1. 引言&#xff1a;流式数据处理的挑战 在现代 AI 应用开发中&#xff0c;流式 API&#xff08;如 OpenAI、Claude 等&#xff09;能实时返回分块数据&#xff0c;提升用户体验。但流式…

【实时Linux实战系列】硬件中断与实时性

在实时系统中&#xff0c;硬件中断是系统响应外部事件的关键机制之一。硬件中断允许系统在执行任务时被外部事件打断&#xff0c;从而快速响应这些事件。然而&#xff0c;中断处理不当可能会导致系统延迟增加&#xff0c;影响系统的实时性。因此&#xff0c;优化中断处理对于提…

基于DTLC-AEC与DTLN的轻量级实时语音降噪系统设计与实现

基于DTLC-AEC与DTLN的轻量级实时语音降噪系统设计与实现 1. 引言 在当今的实时通信应用中,语音质量是影响用户体验的关键因素之一。环境噪声和回声会严重降低语音清晰度,特别是在移动设备和嵌入式系统上。本文将详细介绍如何将两种先进的开源模型——DTLC-AEC(深度学习回声…

基于Hadoop与LightFM的美妆推荐系统设计与实现

文章目录有需要本项目的代码或文档以及全部资源&#xff0c;或者部署调试可以私信博主项目介绍总结每文一语有需要本项目的代码或文档以及全部资源&#xff0c;或者部署调试可以私信博主 项目介绍 本项目旨在基于大数据Hadoop平台和机器学习技术&#xff0c;构建一套面向美妆…

notepad++ 多行复制拼接

如何将中文一 一复制到英文后面按住 ALT ,鼠标左键拖动多行选中中文Ctrl C 复制 在英文的第一行结尾处 Ctrl v 粘贴

【前沿技术动态】【AI总结】Spring Boot 4.0 预览版深度解析:云原生时代的新里程碑

Spring Boot 4.0 预览版深度解析&#xff1a;云原生时代的新里程碑 最低 Java 17&#xff0c;原生支持虚拟线程&#xff0c;性能提升最高800%&#xff0c;Spring Boot 4.0 带来开发体验与运行时性能的全面飞跃 Spring Boot 4.0 的预览版在2025年5月底悄然上线&#xff0c;标志着…

OkHttp 框架封装一个 HTTP 客户端,用于调用外部服务接口

✅ 背景与需求 需要基于 OkHttp 框架封装一个 HTTP 客户端&#xff0c;用于调用外部服务接口&#xff08;如拼团回调&#xff09;&#xff0c;实现以下功能&#xff1a; 动态传入请求地址&#xff08;URL&#xff09;支持 JSON 请求体实现类放在 infrastructure 层的 gateway…

使用Collections.max比较Map<String, Integer>中的最大值

文章目录使用Collections.max比较Map<String, Integer>中的最大值基本方法1. 比较Map的值2. 比较Map的键自定义比较器1. 按值降序排列2. 复杂比较逻辑完整示例代码性能考虑替代方案1. 使用Stream API (Java 8)2. 手动遍历实际应用场景注意事项总结使用Collections.max比较…

鸿蒙状态栏操作

1.鸿蒙设备基础信息 1.1图解 1.1窗口内容规避区域 AvoidArea7 窗口内容规避区域。 窗口内容规避区域。如系统栏区域、刘海屏区域、手势区域、软键盘区域等与窗口内容重叠时&#xff0c;需要窗口内容避让的区域。在规避区无法响应用户点击事件。 除此之外还需注意规避区域的如…