游乐游手机版
首页/AI教程/文章详情

Redis哨兵集群模式部署与SpringBoot架构下应用实践指南

时间:2026-06-16 15:50
基于Redis哨兵模式的高可用方案,通过监控与选举机制实现故障自动转移。文章部署了多个哨兵节点并配置关键参数,模拟主节点宕机后验证了自动切换功能,同时指出重启节点需配置masterauth密码才能正常加入集群。最后介绍了Springboot集成哨兵集群的配置方法,包括yml文件和RedisTemplate封装。

导航

主从集群、哨兵集群、分区集群

环境:

Redis 版本:5.0.8,Spring Boot 版本:2.2.3.RELEASE。Redis 安装指南可参考《CentOS7 下安装 Redis(单机版)》,主从部署详情见《Redis 集群部署及 Spring Boot 架构下应用(主从集群模式)》。

谈到 Redis 的高可用方案,哨兵模式无疑是一个核心话题。今天我们将从部署到 Spring Boot 集成,彻底掌握哨兵集群的实现细节。

集群方式及配置

基础集群配置如下:

后台运行(守护进程)daemonize yes 去除保护模式(允许远程访问)protected-mode no#去除绑定(远程访问)#bind 127.0.0.1# 设置密码requirepass redispwd

本次配置基于同一机器的不同端口构建集群。若 Redis 位于不同机器,部分配置可省略(以下配置过程中会标明【单机非必须】:当前机器仅运行一个 Redis 实例)。

Sentinel 哨兵集群

哨兵模式本质上是一种“监控 + 选举”机制,用于解决主从复制集群中节点宕机后无法自动故障转移的问题。简单来说,它相当于为主从架构配备了一个自动运维团队。

主从配置(略) 参见《Redis 集群部署及 Spring Boot 架构下应用(主从集群模式)》。 哨兵 16379 配置 复制一份配置文件,命名为 sentinel16379.conf(来源为安装目录:redis-5.0.8/sentinel.conf),修改以下配置:

#哨兵服务端口,注意不能跟redis服务端口冲突,且每个哨兵端口不一样port 16379#后台运行(守护进程)daemonize yes#pid写入路径pidfile /var/run/redis-sentinel16379.pid#log目录logfile "./sentinel/16379.log"#工作目录dir ./sentinel/16379#关闭保护模式(取消该行前注释即可)protected-mode no#设置监控主节点信息(参考如下配置)sentinel monitor mymaster 192.168.1.17 6379 2#设置redis密码sentinel auth-pass mymaster redispwd

哨兵 16380 配置 复制 sentinel16379.conf 重命名为 sentinel16380.conf,修改以下配置:

#哨兵服务端口,注意不能跟redis服务端口冲突,且每个哨兵端口不一样port 16380#pid写入路径pidfile /var/run/redis-sentinel16380.pid#log目录logfile "./sentinel/16380.log"#工作目录dir ./sentinel/16380

哨兵 16381 配置 复制 sentinel16379.conf 重命名为 sentinel16381.conf,修改以下配置:

#哨兵服务端口,注意不能跟redis服务端口冲突,且每个哨兵端口不一样port 16381#pid写入路径pidfile /var/run/redis-sentinel16381.pid#log目录logfile "./sentinel/16381.log"#工作目录dir ./sentinel/16381

哨兵 16382 配置 复制 sentinel16379.conf 重命名为 sentinel16382.conf,修改以下配置:

#哨兵服务端口,注意不能跟redis服务端口冲突,且每个哨兵端口不一样port 16382#pid写入路径pidfile /var/run/redis-sentinel16382.pid#log目录logfile "./sentinel/16382.log"#工作目录dir ./sentinel/16382
注意:

这里有一个关键点:sentinel monitor mymaster 192.168.1.17 6379 2 中的“2”代表最低通过票数。也就是说,当至少 2 个哨兵认为主节点不可用时,才会触发故障转移。因此,哨兵总数不能小于该值,通常建议部署奇数个哨兵。

以下是几个与故障判定和切换相关的参数:

#判定master节点失活时间(单位毫秒)sentinel down-after-milliseconds mymaster 30000#主备切换时,slave最大同时同步个数(1:即主备切换每次一个slave节点向master同步)sentinel parallel-syncs mymaster 1#sentinel故障转移超时时间(单位毫秒)sentinel failover-timeout mymaster 180000

其中,192.168.1.17 为主节点地址,6379 为主节点端口,2 为最低通过票数。

启动哨兵

./bin/redis-sentinel sentinel16379.conf ./bin/redis-sentinel sentinel16380.conf ./bin/redis-sentinel sentinel16381.conf ./bin/redis-sentinel sentinel16382.conf

在这里插入图片描述在这里插入图片描述

查看状态(info):

在这里插入图片描述在这里插入图片描述

可以看到主节点状态、IP 端口、从节点数量、哨兵数量(4个),一切正常。接下来模拟主节点宕机场景。

kill 主节点,查看主备切换

在这里插入图片描述在这里插入图片描述

再次查看状态:

在这里插入图片描述在这里插入图片描述

主节点已切换至 6381,哨兵自动完成选举。查看新的主节点信息:

在这里插入图片描述在这里插入图片描述

验证读写:

在这里插入图片描述在这里插入图片描述

从节点读写:

在这里插入图片描述在这里插入图片描述

重启被 kill 的主节点,查看其状态

在这里插入图片描述在这里插入图片描述

在这里插入图片描述在这里插入图片描述

有趣的现象发生了:重启后的节点虽然识别了主节点信息,但并未自动加入从节点列表。查看主节点信息,它仍然显示为一个从节点。这意味着新启动的节点无法正常加入集群工作。问题出在哪里?

分析一下:该节点因手动 kill 而宕机,网络显然正常。那么,它和一个全新的从节点相比,缺少了什么?回顾主从配置:

#主节点地址(ip 端口)slaveof 192.168.1.17 6379#master节点密码,(打开注释,修改master的密码即可)masterauth redispwd

从 info 信息看,主节点 IP 和端口正确,问题大概率出在密码上。最初的 master 节点配置中,并未配置 masterauth。加上 masterauth 后重启,再看结果:

在这里插入图片描述在这里插入图片描述

查看主节点状态:

在这里插入图片描述在这里插入图片描述

问题解决。这个小插曲提醒我们:密码相关的配置必须前后一致,缺一不可。

Sentinel 哨兵集群在 Spring Boot 中的配置

1. pom 引入

org.springframework.bootspring-boot-starter-data-redis

2. yml 配置(properties 配置文件类型请自行转换)

spring:redis: # 集群(哨兵)模式sentinel:nodes:- 192.168.1.17:16379- 192.168.1.17:16380- 192.168.1.17:16381- 192.168.1.17:16382master: mymaster

3. 封装 RedisTemplate(MyRedisConfig.java)

import com.fasterxml.jackson.annotation.JsonAutoDetect;import com.fasterxml.jackson.annotation.PropertyAccessor;import com.fasterxml.jackson.databind.ObjectMapper;import org.springframework.boot.autoconfigure.data.redis.RedisProperties;import org.springframework.cache.annotation.EnableCaching;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import org.springframework.data.redis.connection.RedisConnectionFactory;import org.springframework.data.redis.connection.RedisNode;import org.springframework.data.redis.connection.RedisSentinelConfiguration;import org.springframework.data.redis.core.RedisTemplate;import org.springframework.data.redis.core.StringRedisTemplate;import org.springframework.data.redis.core.ValueOperations;import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;import org.springframework.data.redis.serializer.StringRedisSerializer;@Configuration@EnableCachingpublic class MyRedisConfig {@Beanpublic RedisSentinelConfiguration redisSentinelConfiguration(RedisProperties redisProperties){ RedisSentinelConfiguration configuration = new RedisSentinelConfiguration(redisProperties.getSentinel().getMaster(),new HashSet<>(redisProperties.getSentinel().getNodes()));// 实际使用过程中发现,yml中的密码并不会赋值到 检查RedisSentinelConfiguration发现nodepasswd仅有get和set方法,需要手动加入configuration.setPassword(redisProperties.getPassword().toCharArray());return configuration;}@Beanpublic RedisTemplate redisTemplate(RedisConnectionFactory redisConnectionFactory) {RedisTemplate redisTemplate = new RedisTemplate<>();redisTemplate.setConnectionFactory(redisConnectionFactory);// 使用Jackson2JsonRedisSerialize 替换默认的jdkSerializeable序列化Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);ObjectMapper objectMapper = new ObjectMapper();objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);jackson2JsonRedisSerializer.setObjectMapper(objectMapper);// 设置value的序列化规则和 key的序列化规则redisTemplate.setKeySerializer(new StringRedisSerializer());redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);redisTemplate.afterPropertiesSet();return redisTemplate;}/** * 对redis字符串类型数据操作 * * @param stringRedisTemplate * @return */@Bean({"valueoperations"})public ValueOperations valueOperations(StringRedisTemplate stringRedisTemplate) {return stringRedisTemplate.opsForValue();}}

4. 测试 Controller(SysController.java)

import com.platform.test.common.exception.BusinessException;import com.platform.test.service.SysService;import com.platform.test.vo.BaseRespVo;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.beans.factory.annotation.Qualifier;import org.springframework.data.redis.core.RedisTemplate;import org.springframework.data.redis.core.ValueOperations;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RestController;import javax.servlet.http.HttpSession;import java.text.SimpleDateFormat;import java.util.Date;import java.util.HashMap;import java.util.concurrent.TimeUnit;@RestController@RequestMapping("/sys")public class SysController {static final Logger logger = LoggerFactory.getLogger(SysController.class);@Autowired@Qualifier("valueoperations")ValueOperations valueOperations;@AutowiredRedisTemplate redisTemplate;@AutowiredSysService sysService;final static String REDIS_TEST_KEY_VALUE = "__REDIS_TEST_KEY_VALUE";@RequestMapping("/health")public BaseRespVo health(HttpSession session) throws BusinessException {SimpleDateFormat sf = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");HashMap data = new HashMap<>();data.put("time-server",sf.format(new Date()));data.put("status-redis",checkRedis());return new BaseRespVo(data);} private boolean checkRedis() {long time = System.currentTimeMillis();if(valueOperations == null){return false;}try {valueOperations.set(REDIS_TEST_KEY_VALUE time,REDIS_TEST_KEY_VALUE,300, TimeUnit.MILLISECONDS);logger.info("写入redis key: "  REDIS_TEST_KEY_VALUE time  " value:" REDIS_TEST_KEY_VALUE);Thread.sleep(100);String value = valueOperations.get(REDIS_TEST_KEY_VALUE time);if (value!=null && REDIS_TEST_KEY_VALUE.equals(value)) {logger.info("读取redis key: "  REDIS_TEST_KEY_VALUE time  " value:" value);return true;}}catch (Exception e){logger.error("redis test exception!",e);}return false;}}

执行结果:

在这里插入图片描述在这里插入图片描述

从节点:

在这里插入图片描述在这里插入图片描述

附:

值得注意的是,在 Spring Boot 集成过程中,我们发现 RedisSentinelConfiguration 并不能自动从 yml 配置中读取密码,需要手动调用 setPassword() 方法。这个细节若不注意,很容易导致应用启动后无法连接哨兵集群。如果你有更好的解决方案,欢迎交流探讨。

在这里插入图片描述在这里插入图片描述

补充一点: 在哨兵模式下,我们无需手动监视每个主节点或从节点。启动时,主从关系由 Redis 自身的主从配置决定,哨兵会自动从当前主节点读取集群信息。当被监视的主节点不可达时,哨兵会根据 sentinel monitor 的配置选举出新的主节点,并自动更新自身配置文件。这正是哨兵模式的核心价值所在。

来源:https://cloud.tencent.com.cn/developer/article/2689574
上一篇开源AI演示文稿PPT生成器项目推荐 下一篇实验室智能识别目标检测数据集(YOLO适用)
本站内容用于信息整理与展示,如有侵权或内容问题请及时联系处理。

相关推荐

补充同频道和同主题内容,方便继续浏览更多相关内容。

同类最新

继续查看同栏目最近更新的文章。

更多
RAG四标融合企业知识资产体系四库协同GEO优化实践
AI教程 · 2026-07-01

RAG四标融合企业知识资产体系四库协同GEO优化实践

生成式AI正在彻底改写信息检索的底层逻辑。传统SEO依赖关键词堆砌和外链建设的策略,在大模型的内容采信规则下已经基本失效。取而代之的,是生成式引擎优化(GEO)。它不再关注外链数量,而是重点衡量你的知识是否结构化、证据链是否坚实、信源是否可靠——这些维度才是RAG(检索增强生成)架构真正看重的核心指

一个普通上班人分享WorkBuddy使用心得与真实体验
AI教程 · 2026-07-01

一个普通上班人分享WorkBuddy使用心得与真实体验

前言 最近我开始使用WorkBuddy——这是腾讯推出的一款AI办公工作台。差不多用了一周时间,趁印象还新鲜,把真实的使用感受记录下来,给还在犹豫的朋友做个参考。不吹不黑,只说实际体验。 初印象:不只是聊天机器人 之前用过不少AI工具,大多数就是个对话框,你问它答,答完就结束了。WorkBuddy不

AI幻觉变真功能实战教程:App Inventor 2视频录制拓展一周开发实录
AI教程 · 2026-07-01

AI幻觉变真功能实战教程:App Inventor 2视频录制拓展一周开发实录

先讲一个颇具戏剧性的开端。 这件事的开端颇显荒诞——有用户前来咨询,称AI Pro版的介绍中提到我们有一款“视频录制拓展”。团队全体成员都感到困惑,翻遍产品列表,发现根本不存在该组件。AI那种“一本正经胡说八道”的能力,这次确实让我们陷入尴尬。 按常理,此事到此便可结束——一句“抱歉,暂时没有这个拓

别再混淆OLAP和SQL-on-Hadoop两者查询本质不同
AI教程 · 2026-07-01

别再混淆OLAP和SQL-on-Hadoop两者查询本质不同

OLAP和SQL-on-Hadoop虽都使用SQL查询数据,但本质不同。SQL-on-Hadoop负责海量数据批量计算与ETL,查询速度秒级至分钟级;OLAP通过预聚合实现毫秒级多维分析,适合BI报表。两者在数据平台分工协作,前者是后厨加工,后者是前台快速服务。

GEO优化深度解析:AI偏好FAQ还是长文内容?
AI教程 · 2026-07-01

GEO优化深度解析:AI偏好FAQ还是长文内容?

在GEO优化中,AI对内容形式无统一偏好:FAQ在简单查询中引用率41%,长文在复杂查询中达58%。内容应基于用户意图选择形式,FAQ适配简单事实类问题,长文建立主题权威,两者互补而非替代。