Redis从入门到精通(十四)Redis分布式缓存(二)Redis哨兵集群的搭建和原理分析

文章目录

    • 前言
    • 5.3 Redis哨兵
      • 5.3.1 哨兵原理
        • 5.3.1.1 集群的结构和作用
        • 5.3.1.2 集群监控原理
        • 5.3.1.3 集群故障恢复原理
      • 5.3.2 搭建哨兵集群
      • 5.3.3 RedisTemplate
        • 5.3.3.1 搭建测试项目
        • 5.3.3.2 场景测试

前言

Redis分布式缓存系列文章:

Redis从入门到精通(十三)Redis分布式缓存(一)RDB和AOF持久化、Redis主从集群的搭建与原理分析

5.3 Redis哨兵

Redis提供了哨兵(Sentinel)机制来实现主从集群的自动故障恢复。

5.3.1 哨兵原理

5.3.1.1 集群的结构和作用

哨兵的结构如图:

哨兵的作用如下:

  • 监控:Sentinel会不断检查master和slave是否按预期工作;
  • 自动故障恢复:如果master故障,Sentinel会将一个slave提升为master。当故障实例恢复后,将作为新的master的slave节点加入集群。
  • 通知:Sentinel充当Redis客户端的服务发现来源,当集群发生故障转移时,会将最新信息推送给Redis的客户端。
5.3.1.2 集群监控原理

Sentinel基于心跳机制监测服务状态,每隔1秒向集群的每个实例发送ping命令:

  • 主观下线:如果某Sentinel节点发现某Redis实例未在规定时间响应,则认为该Redis实例主观下线。
  • 客观下线:若超过指定数量(quorum)的Sentinel都认为该Redis实例主观下线,则该Redis实例客观下线。quorum值最好超过Sentinel实例数量的一半。

5.3.1.3 集群故障恢复原理

一旦发现master故障,Sentinel需要在salve中选择一个作为新的master,选择依据是这样的:

  • 首先会判断slave节点与master节点断开时间长短,如果超过指定值(down-after-milliseconds * 10)则会排除该slave节点;
  • 然后判断slave节点的slave-priority值,越小优先级越高,如果是0则永不参与选举;
  • 如果slave-prority一样,则判断slave节点的offset值,越大说明数据越新,优先级越高;
  • 最后是判断slave节点的运行id大小,越小优先级越高。

当选出一个新的master后,Sentinel会进行角色切换,切换流程是这样的:

  • Sentinel给备选的slave1节点发送slaveof no one命令,让该节点成为master;
  • Sentinel给所有其它slave发送slaveof <ip> <port>命令,让这些slave成为新master的从节点,开始从新的master上同步数据;
  • 最后,Sentinel将故障节点标记为slave,当故障节点恢复后会自动成为新的master的slave节点。

5.3.2 搭建哨兵集群

这里将搭建一个三节点形成的Sentinel集群,来监管之前的Redis主从集群。同样,可以在一台虚拟机上创建3个不同端口(27001、27002、27003)的Sentinel实例,来模拟Sentinel集群。

搭建步骤:

  • 1)在/usr/local/redis目录下创建3个目录,名称分别是s1、s2、s3:

  • 2)分别在s1、s2、s3目录下创建配置文件sentinel.conf,添加下面的内容:
# 端口
port 27001
# 以后台进程启动
daemonize yes
sentinel announce-ip "192.168.146.128"
# 指定主节点信息# mymaster 主节点名称,自定义,任意写# 192.168.146.128 7001 主节点的IP和端口# 2 选举master时的quorum值
sentinel monitor mymaster 192.168.146.128 7001 2
sentinel down-after-milliseconds mymaster 5000
sentinel failover-timeout mymaster 60000
# 文件目录
dir "/usr/local/redis/s1"
# 日志文件目录
logfile "/usr/local/redis/s1/sentinel_27001.log"

s1、s2、s3的配置文件中port、dir、logfile应改成各自对应的,其余配置可保持一致。

  • 3)启动3个Sentinel实例**
/usr/local/bin/redis-sentinel /usr/local/redis/s1/sentinel.conf
/usr/local/bin/redis-sentinel /usr/local/redis/s2/sentinel.conf
/usr/local/bin/redis-sentinel /usr/local/redis/s3/sentinel.conf

启动完成后,可查看Sentinel进程如下:

  • 4)测试哨兵作用

尝试让master节点7001宕机,查看Sentinel 27001的日志:

查看7002的日志:

查看7003的日志:

查看此时Redis集群的状态:

至此,Redis哨兵集群搭建完成,并正常工作。

5.3.3 RedisTemplate

在Sentinel集群监管下的Redis主从集群,其节点的角色会因为自动故障转移而发生变化,Redis的客户端必须感知这种变化,及时更新连接信息。

Spring的RedisTemplate底层利用lettuce实现了节点的感知和自动切换。

5.3.3.1 搭建测试项目

下面搭建一个SpringBoot项目来进行测试,步骤如下:

  • 1)引入依赖
<!--pom.xml--><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId></dependency><dependency><groupId>org.apache.commons</groupId><artifactId>commons-pool2</artifactId></dependency>
</dependencies>
  • 2)编写配置文件
// src/main/resources/application.ymlserver:port: 8082
spring:application:name: redis_learning_clusterredis:# Redis主从集群的主节点host: 192.168.146.128port: 7001password: 123321# Redis哨兵集群配置sentinel:master: mymasternodes:- 192.168.146.128:27001- 192.168.146.128:27002- 192.168.146.128:27003lettuce:pool:max-active: 10max-idle: 10min-idle: 1time-between-eviction-runs: 10sjackson:default-property-inclusion: non_null# 修改日志级别为Debug,这样才能查看到RedisTemplate具体连接了哪个实例
logging:level:root: DEBUG
  • 3)编写主启动类
// com.star.redis.cluster.RedisClusterApp@SpringBootApplication
public class RedisClusterApp {public static void main(String[] args) {SpringApplication.run(RedisClusterApp.class, args);}@Beanpublic LettuceClientConfigurationBuilderCustomizer clientConfigurationBuilderCustomizer() {// 配置读写策略// MASTER 从master节点读取// MASTER_PREFERRED 优先从master节点读取,master不可用才读取replica节点// REPLICA 从slave(replica)节点读取// REPLICA_PREFERRED 优先从slave(replica)节点读取,所有的slave都不可用才读取master节点return clientConfigurationBuilder -> clientConfigurationBuilder.readFrom(ReadFrom.REPLICA_PREFERRED);}}
  • 4)编写Controller类
// com.star.redis.cluster.controller.RedisController@RestController
@RequestMapping("/redis")
public class RedisController {@Resourceprivate StringRedisTemplate stringRedisTemplate;@GetMapping("/get/{key}")public String get(@PathVariable String key) {return stringRedisTemplate.opsForValue().get(key);}@GetMapping("/set/{key}/{value}")public String set(@PathVariable String key, @PathVariable String value) {stringRedisTemplate.opsForValue().set(key, value);return "success";}
}
5.3.3.2 场景测试

首先检查一下当前Redis主从集群的状态:

此时7003实例是master节点,7001、7002实例是slave节点。

场景一:调用/redis/set/name/Rose接口,由日志可知写操作由master节点7003实例完成:

场景二:调用/redis/get/name接口,由日志可知读操作由slave节点7002实例完成:

场景三:如果此时master节点7003宕机了,Sentinel会重新选举一个master,这里选择的是7001:

场景四:再次调用/redis/set/name/Rose接口,由日志可知写操作由master节点7001实例完成,意味着RedisTemplate感知到了主从节点的变化:

场景五:再次调用/redis/get/name接口,由日志可知读操作由slave节点7002实例完成:

至此,Redis哨兵集群搭建完毕,且正常工作。

本节完,更多内容请查阅分类专栏:Redis从入门到精通

感兴趣的读者还可以查阅我的另外几个专栏:

  • SpringBoot源码解读与原理分析(已完结)
  • MyBatis3源码深度解析(已完结)
  • 再探Java为面试赋能(持续更新中…)

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

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

相关文章

欧盟网络安全局:公共数据空间中的个人数据保护设计(下)

三、应用场景分析:健康—医药用途 2020年欧盟发布欧盟医药战略,旨在应对制药行业面临的各种机遇和挑战,以确保欧盟公民对于药品的可获得性、可负担性和可持续性。[4]报告将药品数据空间作为一种可能的手段,旨在支持数据使用者对于药品市场供应情况和药品功效的研究和分析。…

Java复习第十七天学习笔记(转发、重定向,GET,POST),附有道云笔记链接

【有道云笔记】十七 4.3 转发、重定向、Get、POST、乱码 https://note.youdao.com/s/GD5TRksQ 一、转发 转发&#xff1a;一般查询了数据之后&#xff0c;转发到一个jsp页面进行展示 req.setAttribute("list", list); req.getRequestDispatcher("student_lis…

大厂面试:获取字符串的全排列

一、概念 现有一个字符串&#xff0c;要打印出该字符串中字符的全排列。例如输入字符串abc&#xff0c;则打印出由字符a、b、c所能排列出来的所有字符串abc、acb、bac、bca、cab和cba。 可以基于回溯法来解决这个问题。 二、代码 public class Permutation {//输出字符串str的全…

权限修饰符,代码块,抽象类,接口.Java

1&#xff0c;权限修饰符 权限修饰符&#xff1a;用来控制一个成员能够被访问的范围可以修饰成员变量&#xff0c;方法&#xff0c;构造方法&#xff0c;内部类 &#x1f47b;&#x1f457;&#x1f451;权限修饰符的分类 &#x1f9e3;四种作用范围由小到大(private<空着…

SV-704XT 100W网络有源音柱 校园广播音柱

SV-704XT 100W网络有源音柱 一、描述 SV-704XT是深圳锐科达电子有限公司的一款壁挂式网络有源音柱&#xff0c;具有10/100M以太网接口&#xff0c;可将网络音源通过自带的功放和喇叭输出播放&#xff0c;其采用防水设计&#xff0c;功率100W。SV-704XT作为网络广播播放系统的终…

java 将 json 数据转为 java 中的对象

一、准备 json 数据 {"name": "mike","age": 17,"gender": 1,"subject": ["math","english"] }二、对应的java对象 package com.demo.controller;import lombok.Data; import java.util.List;Data pu…

回溯算法先导

撤销当前的操作 使用原因及解决的问题 基本上暴力搜索的问题 适用于 组合问题 [1,2,3,4] 两位数的组合有哪些切割问题 给定字符串,求切割方式使其字串都是回文子串子集问题 求 [1,2,3,4] 的子集排列组合 组合(不强调顺序)棋盘问题 如何理解回溯法 抽象为一个树形结构 回溯…

Python模块pyttsx3添加语音包

查询现有语音包信息:脚本import pyttsx3engine = pyttsx3.init() voices = engine.getProperty(voices) for voice in voices:print("Voice:")print(" - ID: %s" % voice.id)print(" - Name: %s" % voice.name)print(" - Languages: %s&qu…

MySQL 04-EMOJI 表情与 UTF8MB4 的故事

拓展阅读 MySQL View MySQL truncate table 与 delete 清空表的区别和坑 MySQL Ruler mysql 日常开发规范 MySQL datetime timestamp 以及如何自动更新&#xff0c;如何实现范围查询 MySQL 06 mysql 如何实现类似 oracle 的 merge into MySQL 05 MySQL入门教程&#xff0…

产品思维训练 | 熊孩子任性打赏从产品角度有哪些方法可以规避?

本周话题&#xff1a; 抖音回应10岁儿童打赏主播10万&#xff1a;已全额退款。正值特殊时期&#xff0c;小朋友们花费在直播APP中的时间也不少。 对于打赏等行为&#xff0c;当然需要家长加强监督&#xff0c;除此之外&#xff0c;产品方面可以做什么措施&#xff0c;压制住胡…

【JS】获取接口返回 EventStream 结构的数据(即接收读取 stream 流)

文章目录 EventStream 是一种服务器推送的数据格式&#xff0c;可以用于实时数据传输。 接口返回的示例图 获取示例&#xff1a; // 这里的 url 为虚拟的&#xff0c;仅供演示用 fetch(https://test.cn.com/api/agent/2, {method: POST,headers: {Content-Type: applicatio…

提取图片地理位置

引言 在数字化时代&#xff0c;图片已经成为我们生活中不可或缺的一部分。然而&#xff0c;如何从图片中提取有用的信息&#xff0c;尤其是地址信息&#xff0c;一直是一个具有挑战性的问题。Python作为一种强大的编程语言&#xff0c;为我们提供了丰富的工具和库来解决这个问…

【SGDR】《SGDR:Stochastic Gradient Descent with Warm Restarts》

arXiv-2016 code: https://github.com/loshchil/SGDR/blob/master/SGDR_WRNs.py 文章目录 1 Background and Motivation2 Related Work3 Advantages / Contributions4 Method5 Experiments5.1 Datasets and Metric5.2 Single-Model Results5.3 Ensemble Results5.4 Experiment…

智慧污水井物联网远程监控案例

智慧污水井物联网远程监控案例 在当今数字化转型的浪潮中&#xff0c;智慧水务已成为城市基础设施建设的重要组成部分。其中&#xff0c;基于物联网技术的智慧污水井远程监控系统以其高效、精准、实时的特性&#xff0c;在提升污水处理效能、保障城市水环境安全、实现精细化管…

每日一题 — 水果成篮

思路&#xff1a; 通过阅读上面文字得出问题&#xff1a;就去只有两个种类的最大长度的连续子数组&#xff0c;这时我们可以想到用哈希表来存储数据&#xff0c;记录数据的种类和每个种类的数量。 解法一&#xff1a;暴力递归&#xff08;right每次遍历完都回退&#xff09; 解…

oceanbase一键安装

安装文档&#xff1a;https://www.oceanbase.com/docs/common-oceanbase-database-cn-1000000000642554 软件下载 https://www.oceanbase.com/softwarecenter 安装obd yum install -y yum-utils yum-config-manager --add-repo https://mirrors.aliyun.com/oceanbase/OceanBa…

无线游戏手柄的测试(Windows11系统手柄调试方法)

实物 1、把游戏手柄的无线接收器插入到电脑usb接口中 2、【控制面板】----【查看设备和打印机】 3、【蓝牙和其它设备】--【更多设备和打印机设置】 4、鼠标右键【游戏控制器设置】 5、【属性】 6、【测试】&#xff08;每个按键是否正常&#xff09; 7、【校准】&#xff08;…

稀碎从零算法笔记Day46-LeetCode:互质树

这几天有点懈怠了 题型&#xff1a;树、DFS、BSF、数学 链接&#xff1a;1766. 互质树 - 力扣&#xff08;LeetCode&#xff09; 来源&#xff1a;LeetCode 题目描述 给你一个 n 个节点的树&#xff08;也就是一个无环连通无向图&#xff09;&#xff0c;节点编号从 0 到 …

从“黑箱”到“透明”:云里物里电子标签助力汽车总装数字化转型

“汽车总装”指“汽车产品&#xff08;包括整车及总成等&#xff09;的装配”&#xff0c;是把经检验合格的数以百计、或数以千计的各种零部件按照一定的技术要求组装成整车及发动机、变速器等总成的工艺过程&#xff0c;是汽车产品制造过程中最重要的工艺环节之一。 其中&…

算法 囚犯幸存者

题目 主类 public static List<Prisoner> prisoners new ArrayList<Prisoner>(); public static List<Prisoner> remainPrisoners new ArrayList<Prisoner>(); public static Prisoner lastPrisoner null;public static void main(String[] args) …