Redis最佳实践——性能优化技巧之集群与分片
Redis集群与分片在电商应用中的性能优化技巧
一、Redis集群架构模式解析
1. 主流集群方案对比
方案 | 核心原理 | 适用场景 | 电商应用案例 |
---|---|---|---|
主从复制 | 读写分离+数据冗余 | 中小规模读多写少 | 商品详情缓存 |
Redis Sentinel | 自动故障转移+监控 | 高可用需求场景 | 订单状态缓存 |
Redis Cluster | 原生分布式分片 | 大规模数据/高并发 | 购物车/秒杀系统 |
代理分片(Twemproxy) | 中间件统一分片 | 兼容旧客户端 | 历史系统改造 |
客户端分片(Sharding) | 客户端计算路由 | 定制化分片策略 | 用户会话管理 |
2. Redis Cluster核心原理
graph TBA[客户端] --> B{CRC16(key) % 16384}B -->|Slot 5500| C[节点A]B -->|Slot 12000| D[节点B]B -->|Slot 3000| E[节点C]C --> F[主节点A1]C --> G[从节点A2]D --> H[主节点B1]D --> I[从节点B2]E --> J[主节点C1]E --> K[从节点C2]
关键机制:
- 数据分片:16384个哈希槽
- Gossip协议:节点间状态同步
- MOVED重定向:客户端自动路由
- ASK重定向:迁移中的临时处理
二、Java客户端集成实践
1. JedisCluster配置示例
public class RedisClusterConfig {@Beanpublic JedisCluster jedisCluster() {Set<HostAndPort> nodes = new HashSet<>();nodes.add(new HostAndPort("10.0.0.1", 7000));nodes.add(new HostAndPort("10.0.0.2", 7000));nodes.add(new HostAndPort("10.0.0.3", 7000));JedisPoolConfig poolConfig = new JedisPoolConfig();poolConfig.setMaxTotal(200);poolConfig.setMaxIdle(50);poolConfig.setTestOnBorrow(true);return new JedisCluster(nodes, 5000, 5000, 5, "password", poolConfig);}
}// 使用示例
public Product getProduct(String id) {try (JedisCluster jedis = jedisCluster.getResource()) {String json = jedis.get("product:" + id);return objectMapper.readValue(json, Product.class);}
}
2. Lettuce高级配置
@Bean(destroyMethod = "shutdown")
public RedisClusterClient redisClusterClient() {List<RedisURI> nodes = new ArrayList<>();nodes.add(RedisURI.create("redis://10.0.0.1:7000"));nodes.add(RedisURI.create("redis://10.0.0.2:7000"));return RedisClusterClient.create(nodes);
}@Bean(destroyMethod = "close")
public StatefulRedisClusterConnection<String, String> clusterConnection() {return redisClusterClient().connect();
}@Bean
public RedisAdvancedClusterCommands<String, String> redisCommands() {return clusterConnection().sync();
}
三、分片策略深度优化
1. 基础分片算法
// CRC16分片算法
public class ShardUtil {public static int getSlot(String key) {return JedisClusterCRC16.getSlot(key);}public static String getShardKey(String prefix, String key, int shards) {int slot = getSlot(key);return prefix + ":" + (slot % shards) + ":" + key;}
}// 使用示例
String productKey = ShardUtil.getShardKey("product", "1001", 16);
jedis.set(productKey, productJson);
2. 热点数据分片优化
// 热点Key检测与动态分片
public class HotKeyProcessor {private static final int HOT_THRESHOLD = 1000; // 每分钟访问量@Scheduled(fixedRate = 60000)public void handleHotKeys() {Map<String, Long> keyStats = getKeyAccessStats();keyStats.entrySet().stream().filter(e -> e.getValue() > HOT_THRESHOLD).forEach(e -> splitHotKey(e.getKey()));}private void splitHotKey(String originalKey) {int shards = calculateOptimalShards(originalKey);migrateData(originalKey, shards);}
}
3. 跨分片事务处理
// 使用Lua脚本实现跨分片原子操作
public boolean crossShardUpdate(String key1, String key2) {String script = "local v1 = redis.call('GET', KEYS[1])\n" +"local v2 = redis.call('GET', KEYS[2])\n" +"if v1 and v2 then\n" +" redis.call('SET', KEYS[1], ARGV[1])\n" +" redis.call('SET', KEYS[2], ARGV[2])\n" +" return 1\n" +"else\n" +" return 0\n" +"end";List<String> keys = Arrays.asList(key1, key2);List<String> args = Arrays.asList("newValue1", "newValue2");Object result = jedis.eval(script, keys, args);return result.equals(1L);
}
四、性能调优参数配置
1. 服务端关键配置
# redis-cluster.conf
cluster-enabled yes
cluster-node-timeout 15000
cluster-migration-barrier 1
cluster-require-full-coverage no
cluster-slave-validity-factor 10# 内存优化
hash-max-ziplist-entries 512
zset-max-ziplist-entries 128
activerehashing yes
2. 客户端连接池配置
GenericObjectPoolConfig<Connection> poolConfig = new GenericObjectPoolConfig<>();
poolConfig.setMaxTotal(500); // 最大连接数
poolConfig.setMaxIdle(100); // 最大空闲连接
poolConfig.setMinIdle(20); // 最小空闲连接
poolConfig.setMaxWaitMillis(200); // 获取连接最大等待时间
poolConfig.setTestOnBorrow(true); // 获取连接时验证
poolConfig.setTestWhileIdle(true); // 空闲连接定期验证
3. 集群监控指标
指标 | 监控命令 | 告警阈值 |
---|---|---|
集群健康状态 | CLUSTER INFO | cluster_state != ok |
分片负载均衡度 | CLUSTER SLOTS | 节点差异 >20% |
迁移状态 | CLUSTER NODES | 迁移中的槽位 >0 |
每秒请求量 | redis-cli --stat | >10万/秒 |
五、实战案例:电商秒杀系统分片设计
1. 库存分片方案
public class InventorySharding {private static final int SHARDS = 32;// 初始化库存分片public void initStock(long productId, int totalStock) {int stockPerShard = totalStock / SHARDS;try (JedisCluster jedis = jedisCluster.getResource()) {for (int i = 0; i < SHARDS; i++) {String key = "stock:" + productId + ":" + i;jedis.set(key, String.valueOf(stockPerShard));}}}// 扣减库存public boolean reduceStock(long productId, String userId) {int shard = userId.hashCode() % SHARDS;String key = "stock:" + productId + ":" + shard;String script = "local current = tonumber(redis.call('GET', KEYS[1]))\n" +"if current > 0 then\n" +" redis.call('DECR', KEYS[1])\n" +" return 1\n" +"end\n" +"return 0";Long result = (Long) jedis.eval(script, Collections.singletonList(key), Collections.emptyList());return result == 1L;}
}
2. 订单号生成分片
public class OrderIdGenerator {private static final int SHARDS = 16;public String generateOrderId(long userId) {int shard = (int) (userId % SHARDS);String key = "order_id:" + shard;Long sequence = jedis.incr(key);return String.format("O%02d%015d", shard, sequence);}
}
六、扩容与迁移方案
1. 在线扩容流程
2. 数据迁移命令
# 将槽位5500从源节点迁移到目标节点
redis-cli --cluster reshard \--cluster-from source_node_id \--cluster-to target_node_id \--cluster-slots 5500 \--cluster-yes
3. Java自动扩容实现
public class AutoScalingManager {@Scheduled(fixedRate = 600000) // 每10分钟检查public void checkClusterStatus() {ClusterInfo clusterInfo = getClusterInfo();if (clusterInfo.getMemoryUsage() > 0.8) {addNewNode();rebalanceCluster();}}private void rebalanceCluster() {List<RedisNode> nodes = getAllNodes();int totalSlots = 16384;int slotsPerNode = totalSlots / nodes.size();// 重新分配槽位for (RedisNode node : nodes) {int targetSlots = slotsPerNode;migrateSlots(node, targetSlots);}}
}
七、故障处理与容灾
1. 脑裂问题解决方案
public class SplitBrainDetector {@Scheduled(fixedRate = 5000)public void checkQuorum() {int liveNodes = getActiveNodeCount();if (liveNodes < (TOTAL_NODES/2 + 1)) {triggerFailSafeMode();}}private void triggerFailSafeMode() {// 1. 停止接受写请求// 2. 记录异常状态// 3. 触发管理员告警}
}
2. 数据恢复流程
八、性能测试数据
1. 集群扩展性测试
节点数 | 吞吐量(QPS) | 平均延迟(ms) | 数据分布均衡度 |
---|---|---|---|
3 | 85,000 | 2.1 | 92% |
6 | 162,000 | 1.8 | 89% |
12 | 305,000 | 1.5 | 85% |
2. 分片策略对比
策略 | 热点处理能力 | 扩容复杂度 | 数据一致性 |
---|---|---|---|
哈希分片 | 中 | 低 | 强 |
范围分片 | 低 | 高 | 强 |
动态分片 | 高 | 中 | 最终一致 |
九、最佳实践总结
-
分片设计原则:
- 将相关数据放在同一分片(如用户所有数据)
- 避免单个分片超过16GB内存
- 预留20%容量缓冲
-
集群管理要点:
- 使用自动化运维工具(如RedisInsight)
- 定期执行
CLUSTER CHECK
命令 - 监控慢查询日志
-
客户端优化:
- 配置合理的连接池参数
- 实现自动重试机制
- 本地缓存热点数据
-
典型问题处理:
// 处理MOVED重定向 public Object handleMoved(JedisCluster jc, String key) {int retry = 0;while (retry++ < 3) {try {return jc.get(key);} catch (JedisMovedDataException e) {refreshClusterInfo();}}throw new RedisException("Max retries exceeded"); }
十、未来扩展方向
-
混合存储架构:
-
AI驱动的弹性扩展:
- 基于预测模型自动调整分片
- 智能预分片算法
- 自动故障预测
-
云原生集成:
- Kubernetes Operator管理
- Serverless自动伸缩
- 多云集群部署
通过合理运用Redis集群与分片技术,电商系统可实现:
- 线性扩展能力:支持千万级QPS
- 99.999%可用性:自动故障转移
- 毫秒级响应:智能数据分布
- PB级存储:无缝水平扩展
更多资源:
https://www.kdocs.cn/l/cvk0eoGYucWA
本文发表于【纪元A梦】
相关文章:

Redis最佳实践——性能优化技巧之集群与分片
Redis集群与分片在电商应用中的性能优化技巧 一、Redis集群架构模式解析 1. 主流集群方案对比 方案核心原理适用场景电商应用案例主从复制读写分离数据冗余中小规模读多写少商品详情缓存Redis Sentinel自动故障转移监控高可用需求场景订单状态缓存Redis Cluster原生分布式分片…...
vue或者前端适配makedown推荐开源依赖
在 Vue 或前端项目中处理 Markdown 格式,以下是一些推荐的开源依赖和工具,根据需求分类整理: 1. 基础 Markdown 解析与渲染 Vue 专用库 vueuse/markdown VueUse 生态的 Markdown 工具,轻量且集成度高。 适合快速在 Vue 项目中渲…...
打打基础 | 从翻转链表到寄存器、汇编与内存
我作为软件工程师在美国工作了三年,期间接触和浸泡过不少的技术栈,罗列一番的话有 AWS cloud, frontend (React, TypeScript), backend (Django, Springboot, ECS, GraphQL), JVM (Java, Scala, Kotlin), data pipelines (Spark, Snowflake, Prefect, DB…...
深入解析 Dotnet-Boxed.Framework:提升 .NET 开发效率的利器
在现代 .NET 开发中,框架和工具的选择对项目的开发效率和长期维护至关重要。Dotnet-Boxed.Framework 是一个开源框架,旨在简化开发流程,提高生产力。它通过一组实用的工具和自动化功能,帮助开发者快速构建高质量的应用程序。本文将…...

常见相机的ISP算法
常见的ISP算法 3A算法 去雾算法 图像增强算法 图像宽动态算法 图像的电子缩放算法,无极电子缩放 图像降噪算法 相机常见问题 1.相机启动速度问题,启动速度较慢 2.相机扛不住高低温问题 3.相机散热问题问题 4.相机高低温芯片保护掉电 5.相机的成像效果或者…...

2024 CKA模拟系统制作 | Step-By-Step | 8、题目搭建-创建 Ingress
目录 免费获取题库配套 CKA_v1.31_模拟系统 一、题目 二、核心考点 Ingress 资源定义 Ingress Controller 依赖 服务暴露验证 网络层次关系 三、搭建模拟环境 1.创建命名空间 2.安装ingress ingress-nginx-controller 3.创建hello.yaml并部署 四、总结 …...

OldRoll复古胶片相机:穿越时光,定格经典
在数字摄影盛行的今天,复古胶片相机的独特魅力依然吸引着无数摄影爱好者。OldRoll复古胶片相机这款软件,以其独特的复古风格和丰富的胶片滤镜效果,让用户仿佛穿越回了那个胶片摄影的黄金时代。它不仅模拟了胶片相机的操作界面,还提…...

通俗易懂的 JS DOM 操作指南:从创建到挂载
目录 🧩 1. 创建元素:document.createElement / createElementNS 📝 2. 创建文本:document.createTextNode ✏️ 3. 修改文本:node.nodeValue 🗑️ 4. 移除元素:el.removeChild() …...

CSS Day07
1.搭建项目目录 2.网页头部SEO三大标签 3.Favicon图标与版心 (1)Favicon图标 (2)版心 4.快捷导航 5.头部-布局 6.头部-logo 7.头部-导航 8.头部-搜索 9头部-购物车 10.底部-布局 11.底部-服务区域 12.底部-帮助中心 13.底部-版权…...
爬虫框架:scrapy使用心得
文章目录 前言一、scrapy是什么?二、使用步骤1.安装和创建2.请求以及参数3.代理池4.请求错误处理5.采集数据入库6.日志及其他配置 总结 前言 有些时候我们需要采集大量数据时,我们需要程序的运行效率高,当然如果有时候不想写请求代码的时候,这些情况我都…...

RV1126-OPENCV 交叉编译
一.下载opencv-3.4.16.zip到自己想装的目录下 二.解压并且打开 opencv 目录 先用 unzip opencv-3.4.16.zip 来解压 opencv 的压缩包,并且进入 opencv 目录(cd opencv-3.4.16) 三. 修改 opencv 的 cmake 脚本的内容 先 cd platforms/linux 然后修改 arm-gnueabi.to…...

【深度学习】 19. 生成模型:Diffusion Models
Diffusion Models Diffusion Models 简介 Diffusion 模型是一类通过逐步添加噪声并再逆向还原的方式进行图像生成的深度生成模型。其基本流程包括: 前向过程(Forward Process):将真实图像逐步加噪,最终变为高斯噪声…...

JMeter 直连数据库
1.直连数据库的使用场景 1.1 参数化,例如登录使用的账户名密码都可以从数据库中取得 1.2 断言,查看实际结果和数据库中的预期结果是否一致 1.3 清理垃圾数据,例如插入一个用户,它的ID不能相同,在测试插入功能后将数据删…...

易路 iBuilder:解构企业 AI 落地困境,重构智能体时代生产力范式
一、从大模型到智能体的产业跃迁 2024 年堪称中国人工智能产业的 "战略拐点" 之年。当 DeepSeek R1 模型以 "技术 价格" 双重普惠模式掀起行业震荡时,各企业纷纷意识到,大模型的真正价值不在于技术炫技,而在于成为企业…...
数据库,Spring Boot,数据源
您是对的,我之前的回答解释了Spring Boot在操作MySQL时不一定需要显式配置指定的数据源类型,因为它有自动配置机制,但没有直接点明在自动配置情况下“数据源是什么”。 在Spring Boot自动配置机制下,这个“数据源”指的是一个连接…...
Linux 第三阶段课程:数据库基础与 SQL 应用
数据库定义 数据库是按一定规则存储数据的仓库,可存储海量数据(百万级至亿级),数据来源包括文本、图像、音视频等多种形式,如出行记录、消费数据等。 数据库分类 关系型数据库:基于二维表格模型࿰…...

计算机网络之路由表更新
1.解题思路 对新接收到的路由表进行更新,全部"距离"1,且"下一跳路由器"都写成发送方路由器的名称。 开始对比新表和原来的路由表 1.看目的网络 如果是新的目的网络,则直接把对应的各项信息填入表中;如果是相同…...

万兴PDF手机版
万兴PDF手机版(万兴PDF编辑器)是一款国产PDF编辑工具.万兴PDF安卓版提供PDF文档编辑,AI撰写摘要,文档签名,设置密码保护等功能,万兴PDF专家APP以简约风格及文档编辑功能为核心,支持多设备终端同步保存.全免 万兴 PDF 编辑器是一款功能强大的 PDF 编辑软件,它支持多种…...

Qt -使用OpenCV得到SDF
博客主页:【夜泉_ly】 本文专栏:【暂无】 欢迎点赞👍收藏⭐关注❤️ 目录 cv::MatdistanceTransform获得SDF 本文的目标, 是简单学习并使用OpenCV的相关函数, 并获得QImage的SDF(Signed Distance Field 有向距离场) 至…...
Python 中Vector类的格式化实现,重点拆解其超球面坐标系的设计精髓
📌 高维向量的格式化革新 在Vector类第5版中,格式化系统迎来重要升级: 坐标系转型:从Vector2d的极坐标(p’后缀)升级为超球面坐标(h’后缀),支持n维空间维度突破&#…...

DDR5 ECC详细原理介绍与基于协议讲解
本文篇幅较长,涉及背景原理介绍方便大家理解其运作方式 以及 基于DDR5协议具体展开介绍。 背景原理介绍 上图参考:DDR 内存中的 ECC 写入操作时,On-die ECC的工作过程如下: SoC将需要写入到Memory中的数据发送给控制器控制器将需要写入的数据直接发送给DRAM芯片在DDR5 DR…...
Linux系统之gettext详解
gettext 是一个用于国际化(i18n)和本地化(l10n)的工具集,旨在帮助开发者创建多语言支持的应用程序。它主要通过提供一系列工具和库来简化文本翻译过程。 gettext 工作流程 标记源代码:在源代码中用 _() 函…...
基于Qt封装数据库基本增删改查操作,支持多线程,并实现SQLite数据库单例访问
抽出来的,直接用就行 头文件CPP文件使用示例 头文件 #ifndef DATABASECOMMON_H #define DATABASECOMMON_H/** 单例封装SQLite通用操作,支持多线程调用;可扩展兼容其他数据库,照着SysRunDatabase写,并且重载openDataba…...

EC800X QuecDuino开发板介绍
支持的模组列表 EG800KEC800MEC800GEC800E 功能列表 基本概述 EC800X QuecDuino EVB 搭载移远 EC800 系列模组。支持模组型号为: EC800M 系列、EC800K 系列、EG800K 系列、EC800E 系列等。 渲染图 开发板的主要组件、接口布局见下图 资料下载 EC800X-QuecDui…...
ARM P15协处理器指令详解:架构、编程与应用实践
ARM P15协处理器指令详解:架构、编程与应用实践 引言 在ARM处理器架构中,协处理器(Coprocessor)系统是扩展处理器功能的关键机制,其中CP15(即协处理器15)作为系统控制协处理器,承担着内存管理、缓存控制、系统配置等核心功能。本…...

PHP轻量级聊天室源码(源码下载)
最新版本:v2.1.2 (2024.08更新) 运行环境:PHP5.6(无需MySQL) 核心特性:手机电脑自适应、TXT数据存储、50条历史消息 适用场景:小型社区/企业内网/教育培训即时通讯 一、核心功能亮点(SEO关键词布…...
MySQL数据表添加字段(三种方式)
基本概念解析 数据表由行和列组成,专业术语中: 字段(Field):表的纵向列结构记录(Record):表的横向行数据 字段添加方法详解 MySQL支持三种字段添加位置,语法格式均通…...

leetcode hot100刷题日记——33.二叉树的层序遍历
解题总结二维vector的初始化方法 题目描述情况1:不确定行数和列数情况2:已知行数和列数情况3:已知行数但不知道列数情况4:已知列数但不知道行数 题目描述 解答:用队列 思路都差不多,我觉得对于我自己来说&a…...
2、PyTorch基础教程:从张量到神经网络训练
1、PyTorch基础 PyTorch 是一个开源的深度学习框架,以其灵活性和动态计算图而广受欢迎。 PyTorch 主要有以下几个基础概念:张量(Tensor)、自动求导(Autograd)、神经网络模块(nn.Module…...

《数据结构初阶》【番外篇:快速排序的前世今生】
【番外篇:快速排序的前世今生】目录 前言:---------------起源---------------一、诞生:二、突破:三、核心: ---------------发展---------------1. 早期版本:简单但不稳定1960 年:初始版本 2. …...