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

news2025/1/12 6:06:30

文章目录

    • 前言
    • 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.yml

server:
  port: 8082
spring:
  application:
    name: redis_learning_cluster
  redis:
    # Redis主从集群的主节点
    host: 192.168.146.128
    port: 7001
    password: 123321
    # Redis哨兵集群配置
    sentinel:
      master: mymaster
      nodes:
        - 192.168.146.128:27001
        - 192.168.146.128:27002
        - 192.168.146.128:27003
    lettuce:
      pool:
        max-active: 10
        max-idle: 10
        min-idle: 1
        time-between-eviction-runs: 10s
  jackson:
    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);
    }

    @Bean
    public 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 {

    @Resource
    private 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.coloradmin.cn/o/1589179.html

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

相关文章

Git分布式版本控制系统——Git常用命令(二)

五、Git常用命令————分支操作 同一个仓库可以有多个分支&#xff0c;各个分支相互独立&#xff0c;互不干扰 分支的相关命令&#xff0c;具体如下&#xff1a; git branch 查看分支 git branch [name] 创建分支&#x…

面试题:重写equals(),为什么还要重写hashcode()

认识equals(): Object类中的equals; public boolean equals(Object obj) {return (this obj);}当我们没有重写equals&#xff08;&#xff09;&#xff0c;我们是调用父类object中的方法&#xff0c;比较的是对象的内存地址 重写equals后&#xff0c; public class Student…

蓝桥杯嵌入式(G431)备赛笔记——DMA+UART

目录 CubeMX配置&#xff1a; 代码配置: DMA通道接收&#xff1a; DMA通道发送&#xff1a; 注意&#xff1a; 主函数中记得开启串口接收回调函数&#xff1a; 加了DMA的UART接收通道和一般的区别&#xff1a; 加了DMA的UART发送和一般的区别&#xff1a; CubeMX配置&…

Day23_学点儿Java_多态复习

1 做错的选择题 Java中的多态性是通过以下哪个机制实现的&#xff1f;&#xff08;&#xff09; A. 方法重载 B. 方法覆盖 C. 抽象类 D. 接口2 多态复习 2.1 学点儿Java_Day7_继承、重载、重写、多态、抽象类 2.2 面向对象四大基本特征 封装、抽象、继承、多态 封装 面向…

【翻译】再见, Clean Code!

&#x1f308;个人主页: 鑫宝Code &#x1f525;热门专栏: 闲话杂谈&#xff5c; 炫酷HTML | JavaScript基础 ​&#x1f4ab;个人格言: "如无必要&#xff0c;勿增实体" 文章目录 【翻译】再见, Clean Code!正文那是一个深夜次日早晨这只是一个阶段 【翻译】再见…

CTF-SHOW SSRF

web351 存在一个flag.php页面&#xff0c;访问会返回不是本地用户的消息&#xff0c;那肯定是要让我们以本地用户去访问127.0.0.1/flag.php web352 代码中先判断是否为HTTP或https协议&#xff0c;之后判断我们传入的url中是否含有localhost和127.0.0&#xff0c;如果没有则…

JAVA集合ArrayList

目录 ArrayList概述 add(element) 用法 add(index, element)用法 remove&#xff08;element&#xff09;用法 remove&#xff08;index&#xff09;用法 get(index)用法 set(index,element) 练习 test1 定义一个集合&#xff0c;添加字符串&#xff0c;并进行遍历&…

【JavaWeb】Day42.MySQL概述——多表查询

介绍 多表查询&#xff1a;查询时从多张表中获取所需数据 单表查询的SQL语句&#xff1a;select 字段列表 from 表名; 如果要执行多表查询&#xff0c;只需要使用逗号分隔多张表即可&#xff0c;如&#xff1a; select 字段列表 from 表1, 表2; 例如&#xff1a;查询…

解决mac本git安装后找不到命令的问题

不熟悉mac配置&#xff0c;折腾了半天&#xff0c;记录一下。 1.问题描述2.解决方法 1.问题描述 从https://sourceforge.net/projects/git-osx-installer/files/下载的git安装包&#xff1a; 安装时提示&#xff1a; 这里的解决办法是按住control键再打开文件安装。 安装完…

数据分析案例(一):地区收入的PCA主成分分析

练习1 地区收入的PCA主成分分析 0.变量说明 1.导包操作 核心思路&#xff1a;导入基础数据操作库包&#xff0c;PCA、k-means 库包&#xff0c;数据可视化库包 import pandas as pd import numpy as np from sklearn.decomposition import PCA from sklearn.preprocessing i…

补体系统研究解决方案BioPorto补体抗体

靶点研究新热点&#xff1a;补体系统 21世纪以来&#xff0c;人们对补体的认识从基于血液的抗菌素系统逐步转变为免疫和组织稳态的全局调节器。近年&#xff0c;对补体激活机制、结构、功能方面的研究更是取得了显著的进展&#xff0c;令补体系统研究跻身热门研究主题之一。 补…

基于SSM新疆旅游管理系统的设计与实现(内附设计LW + PPT+ 源码下载)

摘 要 随着经济的飞速发展人们对于旅游的热衷度也日益提升&#xff0c;新疆因特殊的地理优势具备了天然的旅游资源&#xff0c;天山山脉将新疆的地理区域划分为了南疆和北疆&#xff0c;北疆因气候湿润且河流草场较多造就了得天独厚的自然旅游资源&#xff0c;南疆沙漠广布且…

图像JPEG压缩(附python代码)

JPEG压缩是一种有损压缩技术&#xff0c;常用于数字图像。它通过减少图像文件大小来实现压缩&#xff0c;但会造成一定程度的图像质量损失。 目录 一、JPEG压缩1.1 JPEG压缩原理1.2 JPEG压缩优点1.3 JPEG压缩缺点 二、图像退化中加入JPEG压缩2.1 优势2.1.1 更逼真的模拟2.1.2 提…

OceanMind海睿思助力企业“数据入表”经济利益流入与生命周期管理

通过多年信息系统的建设与应用&#xff0c;企业积累了大量的数据。同时随着时间的推进&#xff0c;数据规模正以加速度快速增长。从国家到企业&#xff0c;都越来越关注所拥有的数据资源及其蕴含的深厚价值。很多企业已经逐渐认知到数据是重要的战略资源&#xff0c;数据资产化…

Acwing.3999 最大公约数(gcd欧拉函数)

题解 给定两个正整数 a,m&#xff0c;其中 a<m。 请你计算&#xff0c;有多少个小于 m 的非负整数 x满足&#xff1a; gcd(a,m)gcd(ax,m) 输入格式 第一行包含整数 T &#xff0c;表示共有 T 组测试数据。 每组数据占一行&#xff0c;包含两个整数 a,m 。 输出格式 每…

2024年第十四届MathorCup数学应用挑战赛C题解析(更新中)

2024年第十四届MathorCup数学应用挑战赛C题解析&#xff08;更新中&#xff09; 题目题目解析(更新中&#xff09;问题一问题二问题三 题目 C题 物流网络分拣中心货量预测及人员排班电商物流网络在订单履约中由多个环节组成&#xff0c;图1是一个简化的物流 网络示意图。其中&a…

UI自动化测试案例

备注:本文为博主原创文章,未经博主允许禁止转载。如有问题,欢迎指正。 个人笔记(整理不易,有帮助,收藏+点赞+评论,爱你们!!!你的支持是我写作的动力) 笔记目录:笔记本~笔记目录_airtest和selenium那个好用-CSDN博客 个人随笔:工作总结随笔_8、以前工作中都接触过哪…

Ansys Zemax | 如何将光栅数据从Lumerical导入至OpticStudio(下)

附件下载 联系工作人员获取附件 本文介绍了一种使用Ansys Zemax OpticStudio和Lumerical RCWA在整个光学系统中精确仿真1D/2D光栅的静态工作流程。将首先简要介绍方法。然后解释有关如何建立系统的详细信息。 本篇内容将分为上下两部分&#xff0c;上部将首先简要介绍方法工作…

如何处理ubuntu22.04LTS安装过程中出现“Daemons using outdated libraries”提示

Ubuntu 22.04 LTS 中使用命令行升级软件或安装任何新软件时&#xff0c;您可能收到“Daemons using outdated libraries”&#xff0c;“Which services should be restarted?”的提示&#xff0c;提示下面列出备选的重启服务&#xff0c;如下。 使用以下命令&#xff0c;能够…

【C】动态规划 之 多维最大最小路径和

总结一下这类题型的思路&#xff1a; 每一步所求的最优解 上一步的最优解 这一步的情况 1.数字三角形 主要思路&#xff1a; 1.到达每一个位置的最大和等于前一步最大和加上这一位置的值&#xff0c;而前一步要么是从左上下来&#xff0c;要么是从右上下来&#xff0c;这样…