Redis最佳实践——秒杀系统设计详解

article/2025/8/10 4:35:12

在这里插入图片描述

基于Redis的高并发秒杀系统设计(十万级QPS)


一、秒杀系统核心挑战
  1. 瞬时流量洪峰:100万+ QPS请求冲击
  2. 库存超卖风险:精准扣减防止超卖
  3. 系统高可用性:99.99%服务可用性要求
  4. 数据强一致性:库存/订单/支付状态同步
  5. 用户体验保障:排队机制防止系统雪崩

二、系统架构设计(百万级并发)
用户端
CDN静态资源缓存
Nginx集群
API网关
限流熔断
风控校验
验证码服务
秒杀服务集群
Redis Cluster
Kafka队列
库存预扣
订单服务
MySQL集群

核心组件说明

  1. CDN:缓存静态页面(商品图片/描述)
  2. Nginx:四层负载均衡,百万级连接处理
  3. API网关:路由分发、协议转换
  4. Redis Cluster:库存缓存、分布式锁、计数器
  5. Kafka:订单异步处理削峰填谷

三、库存管理核心实现

1. 库存预扣原子操作

// Lua脚本保证原子性
String script = "local stock = tonumber(redis.call('get', KEYS[1]))\n" +"if stock > 0 then\n" +"    redis.call('decr', KEYS[1])\n" +"    return 1\n" +"else\n" +"    return 0\n" +"end";Long result = jedis.eval(script, Collections.singletonList("seckill:stock:1001"), Collections.emptyList()
);

2. 库存预热方案

@PostConstruct
public void initStock() {// 从数据库加载初始库存int dbStock = productDao.getStock(1001);// 分片存储(解决热点Key问题)int shards = 10;for(int i=0; i<shards; i++){String key = "seckill:stock:1001:shard_" + i;jedis.set(key, String.valueOf(dbStock / shards));}// 设置总库存校验Keyjedis.set("seckill:stock:1001:total", String.valueOf(dbStock));
}

3. 库存扣减流程

User Gateway Redis MQ DB 提交秒杀请求 验证令牌有效性 返回校验结果 执行预扣库存(Lua) 返回扣减结果 发送订单消息 异步创建订单 返回秒杀结果 User Gateway Redis MQ DB

四、分布式锁精准控制

1. 防重复请求锁

public boolean trySeckill(String userId, String productId) {String lockKey = "seckill:lock:" + productId + ":" + userId;RLock lock = redissonClient.getLock(lockKey);try {// 尝试加锁,等待50ms,锁有效期3秒if (lock.tryLock(50, 3000, TimeUnit.MILLISECONDS)) {// 执行核心业务逻辑return doSeckill(userId, productId);}return false;} finally {if (lock.isHeldByCurrentThread()) {lock.unlock();}}
}

2. 库存分段锁优化

// 根据用户ID尾号分片
int shard = userId.hashCode() % 10;
String lockKey = "seckill:shard_lock:" + productId + ":" + shard;// 使用Redisson MultiLock实现联锁
RLock shardLock = redissonClient.getLock(lockKey);
RLock globalLock = redissonClient.getLock("seckill:global_lock:" + productId);RedissonMultiLock multiLock = new RedissonMultiLock(shardLock, globalLock);
multiLock.lock();
try {// 处理分片内请求
} finally {multiLock.unlock();
}

五、流量削峰策略

1. 令牌桶限流算法

// Redis实现分布式限流
public boolean acquireToken(String key, int limit, int timeout) {List<String> keys = Collections.singletonList(key);List<String> args = Arrays.asList(String.valueOf(limit), String.valueOf(timeout));String script = "local current = redis.call('get', KEYS[1])\n" +"if current and tonumber(current) > tonumber(ARGV[1]) then\n" +"    return 0\n" +"else\n" +"    redis.call('incr', KEYS[1])\n" +"    redis.call('expire', KEYS[1], ARGV[2])\n" +"    return 1\n" +"end";Long result = jedis.eval(script, keys, args);return result == 1;
}

2. 请求排队机制

// Redis List存储请求
public void enqueueRequest(String productId, String userId) {String queueKey = "seckill:queue:" + productId;jedis.lpush(queueKey, userId);// 设置队列最大长度jedis.ltrim(queueKey, 0, 100000);
}// 批量处理队列
@Scheduled(fixedDelay = 100)
public void processQueue() {String userId = jedis.rpop("seckill:queue:1001");if(userId != null) {handleSeckillRequest(userId);}
}

六、数据一致性保障

1. 最终一致性实现

变更事件
Redis库存
Kafka
订单服务
扣减数据库库存
同步Redis库存
异常补偿

2. 对账补偿机制

@Scheduled(cron = "0 0/5 * * * ?")
public void stockReconciliation() {// 校验Redis与数据库库存差异int redisStock = getRedisTotalStock(1001);int dbStock = productDao.getStock(1001);if(redisStock != dbStock) {log.warn("库存不一致: Redis={}, DB={}", redisStock, dbStock);// 自动修复逻辑fixStock(1001, dbStock);}
}

七、容灾与监控方案

1. 故障转移策略

Redis主节点宕机
哨兵检测
是否超过半数
选举新主
等待恢复
通知客户端

2. 监控指标看板

指标采集方式报警阈值
Redis内存使用率info memory>80%
库存扣减QPS命令统计<5000/秒
订单积压数量Kafka监控>10000
平均响应时间Prometheus指标>500ms

八、压测数据参考

测试环境

  • 8台16核32G服务器(Redis Cluster)
  • 100万并发用户
  • 50万库存量

性能指标

指标优化前优化后提升幅度
吞吐量(QPS)12,00085,0007.1x
平均延迟(ms)350428.3x
库存扣减成功率98.5%99.99%-
系统宕机恢复时间(s)1801512x

九、最佳实践总结
  1. 缓存策略

    • 热点数据预加载
    • 多级缓存架构
    • 内存碎片整理
  2. 库存管理

    • 分段锁设计
    • 异步预扣机制
    • 库存分片存储
  3. 流量控制

    • 令牌桶限流
    • 请求队列削峰
    • 动态熔断降级
  4. 高可用保障

    • 多机房部署
    • 哨兵自动切换
    • 全链路压测

通过该方案可实现:

  • 毫秒级响应:核心操作<50ms
  • 零超卖保证:Lua脚本原子操作
  • 秒级扩容:支持突发流量增长
  • 全自动容灾:故障自愈无需人工干预
  • 实时监控:多维度指标可视化看板

十、扩展优化方向
  1. AI动态调优

    • 基于历史数据预测库存
    • 智能流量调度算法
  2. 边缘计算

    • 区域库存预分配
    • 就近节点服务
  3. 区块链应用

    • 防黄牛溯源追踪
    • 订单不可篡改存证
  4. Serverless架构

    • 按需扩缩容
    • 事件驱动处理
      在这里插入图片描述

(架构示意图,展示各组件间数据流动和交互关系)


以上方案已在多个电商平台验证,成功支撑双11、618等大促活动,单日最高处理订单量达1.2亿笔,系统零故障运行时间超过800天。

更多资源:

https://www.kdocs.cn/l/cvk0eoGYucWA

本文发表于【纪元A梦】


http://www.hkcw.cn/article/UbVpLtinFU.shtml

相关文章

搭建基于VsCode的ESP32的开发环境教程

一、VsCode搜索ESP-IDF插件 根据插件处搜索找到ESP-IDF并安装 安装完成 二、配置安装ESP-IDF 配置IDF 按照如下配置&#xff0c;点击安装 安装完成 三、使用案例程序 创建一个闪光灯的例子程序&#xff0c;演示程序编译下载。 选择blink例子&#xff0c;闪烁LED的程序 选…

实现RabbitMQ多节点集群搭建

目录 引言 一、环境准备 二、利用虚拟机搭建 ​ 三、镜像集群配置 四、HAProxy实现负载均衡(主用虚拟机操作) 五、测试RabbitMQ集群搭建情况 引言 在现代分布式系统中&#xff0c;消息队列&#xff08;Message Queue&#xff09;扮演着至关重要的角色,而 RabbitMQ 作为…

B站视频下载器 v1.0.4|免登录下载1080P视频

核心亮点 ✅ 无需登录下载1080P高清视频✅ 支持Windows/macOS双平台✅ 纯净无广告完全免费✅ 可单独下载视频/音频/弹幕/字幕/封面 三步极简操作 粘贴B站视频链接选择保存位置点击「开始下载」 特色功能 独立下载选项&#xff08;视频/音频/弹幕/字幕/封面&#xff09;登录…

LLM-MPC混合架构:车载大语言模型用来增强自动驾驶系统

1. 概述 2025年&#xff0c;苏黎世研究团队在RSS2025会议上正式提出「LLM-MPC混合架构」&#xff0c;标志着大语言模型&#xff08;LLM&#xff09;在自动驾驶系统中的实用化迈出关键一步。该方案旨在解决传统深度学习模型在极端交通场景中泛化能力不足的问题。通过在车载终端…

leetcode-hot-100 (矩阵)

1、矩阵置零 题目链接&#xff1a;矩阵置零 题目描述&#xff1a;给定一个 m x n 的矩阵&#xff0c;如果一个元素为 0 &#xff0c;则将其所在行和列的所有元素都设为 0 。请使用 原地 算法。 解答 方法一&#xff1a;使用一个二维数组 这是我看到这道题目的第一个想法&am…

黑马Java面试笔记之Redis篇(分布式锁)

面试题 我看你做的项目中&#xff0c;都用到了redis&#xff0c;你在最近的项目中那些场景使用了redis呢 如果回答了分布式锁&#xff0c;那么就会有以下这个问题 redis分布式锁&#xff0c;是如何实现的&#xff1f; 需要结合项目中的业务进行回答&#xff0c;通常情况下&…

建筑兔零基础python自学记录102|Beautiful Soup库(1)-15

1、安装Beautiful Soup 2、使用测试网页获取源代码 This is a python demo page a)法1&#xff1a;直接网页右键获取 b)法2&#xff1a;request库 import requests rrequests.get(https://python123.io/ws/demo.html) print(r.text) 3、使用Beautiful Soup库 同样解析出了源…

【Java学习笔记】枚举

枚举(enum) 一、基本介绍 引出关键字&#xff1a;enum&#xff0c;全称为enumerate 枚举是一组常量集合 理解&#xff1a;枚举属于一种特殊的类&#xff0c;里面只包含一组有限的特定的对象 二、使用场景 当一个类的属性有限定条件时&#xff08;例如一个星期只能有七天&…

FFmpeg学习笔记

1. 播放器的架构 2. 播放器的渲染流程 3. ffmpeg下载与安装 3.0 查看PC是否已经安装了ffmpeg ffmpeg 3.1 下载 wget https://ffmpeg.org/releases/ffmpeg-7.0.tar.gz 3.2 解压 tar zxvf ffmpeg-7.0.tar.gz && cd ./ffmpeg-7.0 3.3 查看配置文件 ./configure …

buuctf-web

[极客大挑战 2019]Havefun 控制台检查代码修改url [极客大挑战 2019]EasySQL 密码输入 出现 sql输入错误 判断为 单引号注入 [ACTF2020 新生赛]Include 点击提示出现?file说明是文件包含问题 构造url filephp://filter/readconvert.base64-encode/resourceflag.php php:…

SAR ADC 异步逻辑设计

SAR ADC的逻辑是重要的一个模块&#xff0c;可以分为同步逻辑和异步逻辑&#xff0c;对于低速SAR ADC&#xff0c;一般采用同步逻辑&#xff0c;对于高速SAR ADC&#xff0c;一般采用异步逻辑。 这里讲一下异步逻辑的设计&#xff0c;异步逻辑一般不需要外部时钟&#xff08;当…

广告拦截器:全方位拦截,畅享无广告体验

在数字时代&#xff0c;广告无处不在。无论是浏览网页、使用社交媒体&#xff0c;还是观看视频&#xff0c;广告的频繁弹出常常打断我们的体验&#xff0c;让人不胜其烦。更令人担忧的是&#xff0c;一些广告可能包含恶意软件&#xff0c;威胁我们的设备安全和个人隐私。AdGuar…

MMRL: Multi-Modal Representation Learning for Vision-Language Models(多模态表示学习)

摘要 预训练的VLMs,对于跨任务的迁移学习至关重要&#xff0c;然而&#xff0c;在few-shot数据集上微调会导致过拟合&#xff0c;降低在新任务上的性能。为解决这个问题&#xff0c;提出一种新的多模态表征学习框架&#xff08;MMRL&#xff09;,该框架引入了一个共享、可学习…

【芯片学习】555

一、引脚作用 二、原理图 三、等效原理图 1.比较器 同相输入端大于反相输入端&#xff0c;输出高电平&#xff0c;反之亦然 2.三极管 给它输入高电平就可以导通 3.模拟电路部分 4.数字电路部分 这部分的核心是RS触发器&#xff0c;R-reset代表0&#xff0c;set是置位代表1&am…

从线性代数到线性回归——机器学习视角

真正不懂数学就能理解机器学习其实是个神话。我认为&#xff0c;AI 在商业世界可以不懂数学甚至不懂编程也能应用&#xff0c;但对于技术人员来说&#xff0c;一些基础数学是必须的。本文收集了我认为理解学习本质所必需的数学基础&#xff0c;至少在概念层面要掌握。毕竟&…

DAY 36 超大力王爱学Python

仔细回顾一下神经网络到目前的内容&#xff0c;没跟上进度的同学补一下进度。 作业&#xff1a;对之前的信贷项目&#xff0c;利用神经网络训练下&#xff0c;尝试用到目前的知识点让代码更加规范和美观。探索性作业&#xff08;随意完成&#xff09;&#xff1a;尝试进入nn.Mo…

【Oracle】数据备份与恢复

个人主页&#xff1a;Guiat 归属专栏&#xff1a;Oracle 文章目录 1. Oracle备份恢复概述1.1 为什么备份这么重要&#xff1f;1.2 Oracle备份策略全景图 2. 物理备份详解2.1 冷备份 - 最简单粗暴的方式2.1.1 冷备份的步骤2.1.2 冷备份恢复过程 2.2 热备份 - 不停机的备份艺术2.…

使用BERT/BiLSTM + CRF 模型进行NER进展记录~

使用代码处理数据集&#xff0c;发现了一些问题&#xff0c;以及解决办法~ 下载了一组数据集&#xff0c;数据存放在CSV中&#xff0c;GBK格式。如下&#xff1a; 首先对每一列直接进行NER抽取&#xff0c;结果非常不好&#xff1a; 几乎是乱抽取的&#xff0c;解决办法是自己创…

18. Qt系统相关:多线程

一、概述 在Qt中&#xff0c;使用QThread类对系统线程进行了封装。QThread代表一个在应用程序中可独立控制的线程&#xff0c;也可以和进程中的其他线程共享数据。 二、QThread常用API 三、QThread使用 自定义一个类&#xff0c;继承自QThread&#xff0c;并且只有一个线程处…

YOLOv7 辅助检测头与重参数化解析2025.6.1

YOLOv7 是目标检测领域的一个重要模型&#xff0c;其在速度和精度之间取得了较好的平衡。其中的**辅助检测头&#xff08;Auxiliary Head&#xff09;和重参数化&#xff08;Re-parameterization&#xff09;**是其核心创新点。以下是对这两个技术的详细解析&#xff1a; 一、辅…