从0到1构建高并发秒杀系统:实战 RocketMQ 异步削峰与Redis预减库存

🚀从0到1构建高并发秒杀系统:实战 RocketMQ 异步削峰与Redis预减库存


📖一、 简介

在电商、抢票等高并发场景中,秒杀系统面临着“高并发、库存稀缺、易超卖、系统易崩”的严峻挑战。传统的同步处理架构难以支撑海量请求并发下的性能与一致性要求。

本文从实战出发,系统性地讲解如何基于 Redis + RocketMQ + MySQL + Spring Boot 构建一个高性能、高可用、强一致性的秒杀系统。通过 接口限流、Redis 原子扣减库存、RocketMQ 异步削峰、数据库幂等落库 等机制,彻底解决了高并发下的核心问题,如超卖、重复下单、系统崩溃等。

你将看到:

  • 🧠 秒杀系统面临的本质问题与设计原则
  • 🧱 架构层次的分层与职责划分
  • ⚙️ Redis + Lua 脚本实现库存预扣与并发控制
  • 🚀 RocketMQ 异步下单实现削峰填谷与解耦
  • 💡 MySQL 乐观锁 + 幂等设计实现最终一致性
  • 🛠️ 全程配套详细注释代码、架构图与数据库设计

本文不仅提供了完整可运行的思路,还具备工程级可落地性。适合架构师、后端工程师在面对实际高并发场景时作为参考与实践蓝图。


🧠 二、秒杀系统的挑战与本质

“秒杀”业务场景常出现在电商、抢票、预约系统中,具有以下挑战:

维度问题说明
并发性高并发访问瞬时请求高达数十万甚至上百万
数据一致性超卖/重复下单库存是关键共享资源
系统稳定性容易雪崩单点性能瓶颈可能导致系统挂掉
响应速度秒级反馈用户希望秒杀是否成功即时反馈

核心:限流 + 削峰 + 异步 + 缓存


🔧 三、系统总体架构设计

在这里插入图片描述


🔨 四、核心技术选型与职责

技术组件作用
Redis缓存库存、原子扣减、用户状态标记
RocketMQ削峰填谷、异步解耦
MySQL最终订单存储、库存持久化
Spring Boot微服务框架
Guava RateLimiter接口级限流

📦 五、秒杀系统核心模块详细设计


1️⃣ 接口限流 + 秒杀入口

@RestController
@RequestMapping("/seckill")
public class SeckillController {// 每秒只允许100个请求通过private final RateLimiter rateLimiter = RateLimiter.create(100);@Autowiredprivate SeckillService seckillService;@PostMapping("/{itemId}")public ResponseEntity<String> seckill(@PathVariable Long itemId) {// 通过令牌桶控制请求速率if (!rateLimiter.tryAcquire()) {return ResponseEntity.status(HttpStatus.TOO_MANY_REQUESTS).body("请求过多,请稍后重试");}// 模拟用户获取(实际从登录信息中取)Long userId = 1001L;boolean success = seckillService.processSeckill(itemId, userId);if (success) {return ResponseEntity.ok("请求成功,正在排队中...");} else {return ResponseEntity.badRequest().body("库存不足或已抢完");}}
}

2️⃣ Redis 原子扣减库存(Lua 脚本)+ RocketMQ 消息发送

@Service
public class SeckillServiceImpl implements SeckillService {private static final String STOCK_KEY_PREFIX = "seckill:stock:";@Autowiredprivate RedisTemplate<String, Object> redisTemplate;@Autowiredprivate RocketMQTemplate rocketMQTemplate;@Overridepublic boolean processSeckill(Long itemId, Long userId) {String stockKey = STOCK_KEY_PREFIX + itemId;// Lua 脚本:原子检查库存并扣减String luaScript = "if (tonumber(redis.call('get', KEYS[1])) > 0) then " +" return redis.call('decr', KEYS[1]) " +"else return -1 end";// 执行脚本,防止并发引起库存超卖DefaultRedisScript<Long> redisScript = new DefaultRedisScript<>();redisScript.setScriptText(luaScript);redisScript.setResultType(Long.class);Long result = redisTemplate.execute(redisScript, Collections.singletonList(stockKey));if (result == null || result < 0) {// 库存不足return false;}// 构建订单消息并发送到 MQSeckillOrderMessage message = new SeckillOrderMessage(userId, itemId);rocketMQTemplate.convertAndSend("seckill-topic", message);return true;}
}

3️⃣ RocketMQ 消费者:监听秒杀订单消息,落库处理

@Component
@RocketMQMessageListener(topic = "seckill-topic", consumerGroup = "seckill-consumer-group")
public class SeckillConsumer implements RocketMQListener<SeckillOrderMessage> {@Autowiredprivate SeckillOrderService orderService;@Overridepublic void onMessage(SeckillOrderMessage msg) {try {// 调用下单服务进行库存校验和订单落库orderService.createOrder(msg.getUserId(), msg.getItemId());} catch (Exception e) {// 消息重试或记录异常用于补偿System.err.println("消费失败: " + e.getMessage());}}
}

4️⃣ 下单服务:幂等校验 + 数据库存储 + 乐观锁扣减

@Service
public class SeckillOrderServiceImpl implements SeckillOrderService {@Autowiredprivate SeckillOrderRepository orderRepository;@Autowiredprivate StockRepository stockRepository;@Transactionalpublic void createOrder(Long userId, Long itemId) {// 幂等校验:防止重复下单(可用唯一索引或Redis SET)if (orderRepository.existsByUserIdAndItemId(userId, itemId)) {return;}// 扣减数据库库存,使用乐观锁 versionint updated = stockRepository.decreaseStock(itemId);if (updated == 0) {throw new RuntimeException("库存不足,数据库扣减失败");}// 写入订单记录SeckillOrder order = new SeckillOrder(userId, itemId, LocalDateTime.now());orderRepository.save(order);}
}

5️⃣ 数据库表结构设计(库存 + 订单)

📌 商品库存表(带 version 乐观锁)
CREATE TABLE stock (id BIGINT PRIMARY KEY AUTO_INCREMENT,item_id BIGINT NOT NULL UNIQUE,stock INT NOT NULL,version INT NOT NULL DEFAULT 0
);
📌 库存扣减 SQL(乐观锁)
UPDATE stock
SET stock = stock - 1, version = version + 1
WHERE item_id = ? AND version = ? AND stock > 0;
📌 订单表
CREATE TABLE seckill_order (id BIGINT PRIMARY KEY AUTO_INCREMENT,user_id BIGINT NOT NULL,item_id BIGINT NOT NULL,create_time DATETIME NOT NULL,UNIQUE KEY uniq_user_item (user_id, item_id)
);

📊 六、性能优化与高可用建议

方向建议
限流接口层限流(Guava)、网关限流(Sentinel)
削峰使用 RocketMQ 异步下单,防止数据库击穿
幂等Redis SETNX、唯一索引、分布式锁
日志与监控Prometheus + Grafana、日志追踪链路
高可用RocketMQ 主从部署、Broker 宕机自动切换

✅ 七、总结

通过将 Redis 与 RocketMQ 结合,构建出一个具备如下特性的高并发秒杀系统:

  • 并发控制得当:接口限流 + Redis 原子性操作
  • 系统抗压能力强:消息削峰,MQ异步下单
  • 数据一致性高:数据库落库有幂等保障
  • 用户体验更好:请求秒级响应,后台异步处理

📘 八、后续可拓展点

  • 秒杀结果异步通知(短信 / WebSocket)
  • Redis 秒杀状态标识(用户是否成功)
  • RocketMQ 事务消息提升可靠性
  • 异常消息记录 + 自动补偿机制

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/web/83802.shtml

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

相关文章

OpenCV CUDA模块图像变形------对图像进行任意形式的重映射(Remapping)操作函数remap()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 该函数根据给定的映射表 xmap 和 ymap 对图像进行 任意形式的重映射&#xff08;Remapping&#xff09;操作&#xff0c;是 GPU 加速版本的图像几…

PC 端常用 UI 组件库

一、前言 随着企业级应用、后台管理系统、数据平台等项目的不断发展&#xff0c;前端开发已经不再局限于移动端和响应式布局&#xff0c;而是越来越多地聚焦于 PC 端系统的构建。为了提升开发效率、统一设计风格并保障用户体验&#xff0c;使用成熟的 UI 组件库 成为了现代前端…

pikachu靶场通关笔记31 文件包含02之远程文件包含

目录 一、文件包含功能 二、文件包含Vulnerability 二、远程文件包含 三、环境配置 1、进入靶场 2、搭建环境 &#xff08;1&#xff09;定位php.ini文件 &#xff08;2&#xff09;修改php.ini文件 四、源码分析 五、渗透实战 1、选择科比 2、执行phpinfo &…

QT集成Boost库

在Windows平台上&#xff0c;使用Qt集成Boost库&#xff0c;并基于MSVC编译器在CMake文件中加载&#xff0c;可以按照以下步骤进行配置。 Boost库的编译 如果Boost库未预编译&#xff0c;需要手动编译&#xff0c;解压zip到D:\Library\boost_1_87_0&#xff0c;打开cmd命令行…

MySQL从库复制延迟的监测

目录 ⏱️ 一、原生内置方法❤️ 二、心跳表工具&#xff08;如pt-heartbeat&#xff09;⚙️ 三、MySQL 8.0 增强方案&#x1f4ca; 四、各方案对比总结&#x1f48e; 五、选择建议 MySQL从库复制延迟的监测是保障数据一致性和读写分离可靠性的关键环节&#xff0c;以下是主流…

slam--最小二乘问题--凹凸函数

最小二乘问题 最小二乘问题标准公式 残差函数&#xff0c;线性和非线性最小二乘 最小二乘问题的两种写法&#xff1a; 目标 找到 x使得预测值 Ax与观测值 b 的残差平方和最小。 范数和范数平方 线性最小二乘 一般形式&#xff1a; 残差 rAx−b是x 的线性函数。 目标函数是…

crackme008

crackme008 名称值软件名称Andrnalin.1.exe加壳方式无保护方式serial编译语言Microsoft Visual Basic调试环境win10 64位使用工具x32dbg,PEid破解日期2025-06-10 脱壳 1. 先用PEid查壳 查到无壳 寻找Serial 寻找flag&#xff0c;用x32dbg打开程序&#xff0c;鼠标右键->…

【C语言】图书管理系统(文件存储版)丨源码+详解

一、系统介绍 这是一个基于C语言开发的终端图书管理系统&#xff0c;采用链表数据结构​​二进制文件存储技术实现。系统具有以下特点&#xff1a; ​双角色系统​&#xff1a;管理员&#xff08;管理图书&#xff09;和读者&#xff08;借阅/归还&#xff09;​完整功能​&a…

Java求职者面试题解析:Spring、Spring Boot、MyBatis框架与源码原理

Java求职者面试题解析&#xff1a;Spring、Spring Boot、MyBatis框架与源码原理 第一轮&#xff1a;基础概念问题 1. 请解释什么是Spring框架&#xff1f;它的核心特性有哪些&#xff1f; Spring是一个开源的Java/Java EE应用程序框架&#xff0c;用于简化企业级应用开发。其…

【Chipyard】修改Gemmini 中PE的数量

实战目标 PE数量扩大到原来4倍 原来配置 tileRows: Int 1,tileColumns: Int 1,meshRows: Int 16,meshColumns: Int 16, 改后配置 tileRows: Int 1,tileColumns: Int 1,meshRows: Int 32,meshColumns: Int 32, 修改配置 1. 修改gemmini的scala配置文件&#xff0c;用…

TCP客户端进程分割输入输出

创建TCP客户端&#xff0c;创建子进程分割TCP客户端的read功能和write功能&#xff0c;实现分割I/O的目的。加快客户端速率。 #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <signal.h> #include …

Wi-Fi 6 在 2.4GHz 频段的速率与优化分析

Wi-Fi 6&#xff08;802.11ax&#xff09;在 2.4GHz 和 5GHz 频段均可工作&#xff0c;理论最高速率可达 1200 Mbps&#xff08;5GHz&#xff0c;80MHz&#xff0c;22 MIMO&#xff09;。但在 2.4GHz 频段&#xff0c;速率受 信道宽度、MIMO、调制方式、干扰、协议开销 影响&am…

WPF--Application.Current.Dispatcher.BeginInvoke

1.代码示例 private void LogInfoList_CollectionChanged(object? sender, NotifyCollectionChangedEventArgs e) {// 直接在这里修改集合会引发递归if (e.Action NotifyCollectionChangedAction.Add){if (logInfoList.Count > 200){logInfoList.RemoveAt(0); // 这里会…

ZooKeeper详解以及应用部署(AI)

ZooKeeper 是一个开源的分布式协调服务框架&#xff0c;旨在为分布式应用提供一致性保障和关键协调功能。其核心设计理念是将复杂的分布式一致性逻辑封装为简单可靠的接口&#xff0c;让开发者专注于业务逻辑而非底层协调难题。以下是其核心要点&#xff1a; &#x1f9e0; 一、…

将MySQL数据库中所有表和字段编码统一改为utf8mb4_unicode_ci

完整操作步骤 1. 首先修改数据库默认字符集 sql ALTER DATABASE 你的数据库名 CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci; 2. 生成批量修改所有表的SQL语句 sql SELECT CONCAT(ALTER TABLE , table_schema, ., table_name, CONVERT TO CHARACTER SET utf8mb4 C…

jupyterhub的浅浅使用-重点在解决无法登录

jupyterhub的浅浅使用-重点在解决无法登录 jupyterhub的浅浅使用-重点在解决无法登录1、jupyterhub是什么2、创建Dockerfile3、启动容器3.1、生成配置文件jupyterhub --generate-config3.2、运行容器3.3、进入容器配置用户密码3.4、访问127.0.0.1:8000并登录 4、后台创建的用户…

【Bitcoin基础】比特币的地址格式有哪些?如何应用?

比特币地址格式的分类及应用场景 比特币地址是用于接收和发送比特币的标识符&#xff0c;主要有以下几种格式&#xff0c;每中类型都有其特定的用途和特点: 比特币地址格式 P2PKH 1xxxx leagcy地址 P2SH 3xxxx 允许更复杂的交易多重签名 bech32 bc1xxxx bech32mP2TR…

3.1.2_栈的顺序存储实现

知识总览&#xff1a; 顺序栈的定义&#xff1a; 顺序栈是用顺序存储实现的 &#xff0c;代码定义方式和顺序表类似(啥是顺序表来着&#xff1f;&#xff1f;&#xff1f;) 定义一个顺序栈struct结构体SqStack&#xff0c;结构体中有静态数组data来存放栈里边的元素1个int型的…

JavaEE初阶第一期:计算机是如何 “思考” 的(上)

专栏&#xff1a;JavaEE初阶起飞计划 个人主页&#xff1a;手握风云 一、冯诺依曼体系结构 1.1. 概念 冯诺依曼体系结构&#xff08;Von Neumann Architecture&#xff09;&#xff0c;是现代计算机的基础设计概念&#xff0c;核心思想是“存储程序控制”。具体来说&#xff0c…

SQL Server全局搜索:在整个数据库中查找特定值的高效方法

SQL Server全局搜索&#xff1a;在整个数据库中查找特定值的高效方法 一、需求背景&#xff1a;为什么需要数据库全局搜索&#xff1f; 在数据库管理和开发过程中&#xff0c;我们经常会遇到这样的场景&#xff1a; 只记得某个数据值&#xff0c;但忘记了它所在的表或列需要…