今天来聊聊 Redis 的主从复制

a72a7111b06e16b1225109b49f03cca8.gif

作者 | 阿Q

来源 | 阿Q说代码

今天我们就从配置文件、设计原理、面试真题三个方面来聊一聊 Redis 的主从复制。

在 Redis 复制的基础上,使用和配置主从复制非常简单,能使得从 Redis 服务器(下文称 replica)能精确的复制主 Redis 服务器(下文称 master)的内容。每次当 replica 和 master 之间的连接断开时, replica 会自动重连到 master 上,并且无论这期间 master 发生了什么, replica 都将尝试让自身成为 master 的精确副本。

主从复制,从 5.0.0 版本开始,Redis 正式将 SLAVEOF  命令改名成了 REPLICAOF 命令并逐渐废弃原来的 SLAVEOF 命令

Redis使用默认的异步复制,其特点是低延迟高性能,是绝大多数 Redis 用例的自然复制模式。但是,replica 会异步地确认它从主 master 周期接收到的数据量。

主从拓扑架构

5c0e607a1ab610fa584fe14cf671fb44.png

master 用来写操作,replicas 用来读取数据,适用于读多写少的场景。而对于写并发量较高的场景,多个从节点会导致主节点写命令的多次发送从而过度消耗网络带宽,同时也加重了 master 的负载影响服务稳定性。

a55baaf3c2dd963a55561f92ec5f3abc.png

replica 可以接受其它 replica 的连接。除了多个 replica 可以连接到同一个 master 之外, replica 之间也可以像层叠状的结构(cascading-like structure)连接到其他 replica 。自 Redis 4.0 起,所有的 sub-replica 将会从 master 收到完全一样的复制流。

当 master 需要多个 replica 时,为了避免对 master 的性能干扰,可以采用树状主从结构降低主节点的压力。

为了让大家对概念有更清晰的认识,我们先来看一下配置文件中主从复制的参数介绍:

REPLICATION

replicaof <masterip> <masterport>

通过设置 master 的 ip 和 port ,可以使当前的 Redis 实例成为另一台 Redis 实例的副本。在Redis启动时,它会自动从 master 进行数据同步。

  • Redis 复制是异步的,可以通过修改 master 的配置,在 master 没有与给定数量的 replica 连接时,主机停止接收写入;

  • 如果复制链路丢失的时间相对较短,Redis replica 可以与 master 执行部分重新同步,可以使用合理的 backlog 值来进行配置(见下文);

  • 复制是自动的,不需要用户干预。在网络分区后,replica 会自动尝试重新连接到 master 并与 master 重新同步;

masterauth <master-password>

当 master 设置了密码保护时,replica 服务连接 master 的密码

replica-serve-stale-data yes

当 replica 与 master 失去连接或者主从复制在进行时,replica 可以有两种不同的设置:

  • replica-serve-stale-data:yes(默认值),则 replica 仍将响应客户端请求,可能会有过期数据,或者如果这是第一次同步,则数据集可能为空。

  • replica-serve-stale-data:no , replica 将对所有请求命令(但不包含 INFO, replicaOF, AUTH, PING, SHUTDOWN, REPLCONF, ROLE, CONFIG, SUBSCRIBE, UNSUBSCRIBE, PSUBSCRIBE, PUNSUBSCRIBE, PUBLISH, PUBSUB, COMMAND, POST, HOST: and LATENCY)返回 SYNC with master in progress 的错误。

replica-read-only

可以将 replica 配置为是否只读,yes 代表为只读状态,将会拒绝所有写入命令;no 表示可以写入。从 Redis 2.6 之后, replica 支持只读模式且默认开启。可以在运行时使用 CONFIG SET 来随时开启或者关闭。

对 replica 进行写入可能有助于存储一些临时数据(因为写入 replica 的数据在与 master 重新同步后很容易被删除),计算慢速集或排序集操作并将其存储到本地密钥是多次观察到的可写副本的一个用例。但如果客户端由于配置错误而向其写入数据,则也可能会导致问题。

级联结构中即使 replica B 节点是可写的,Sub-replica C 也不会看到 B 的写入,而是将拥有和 master A 相同的数据集。

设置为 yes 并不表示客户端用集群方式以 replica 为入口连入集群时,不可以进行 set 操作,且 set 操作的数据不会被放在 replica 的槽上,会被放到某 master 的槽上。

注意:只读 replica 设计的目的不是为了暴露于互联网上不受信任的客户端,它只是一个防止实例误用的保护层。默认情况下,只读副本仍会导出所有管理命令,如CONFIG、DEBUG 等。在一定程度上,可以使用rename-command来隐藏所有管理/危险命令,从而提高只读副本的安全性

repl-diskless-sync

复制同步策略:磁盘(disk)或套接字(socket),默认为 no 使用 disk 。

新的 replicas 和重新连接的 replicas 如果因为接收到差异而无法继续复制过程,则需要执行“完全同步”。RDB 文件从 master 传送到 replicas,传输可以通过两种不同的方式进行:

  1. Disk-backed:Redis master 节点创建一个新的进程并将 RDB 文件写入磁盘,然后文件通过父进程增量传输给 replicas 节点;

  2. Diskless:Redis master 节点创建一个新的进程并直接将 RDB 文件写入到 replicas 的 sockets 中,不写到磁盘。

  • 当进行 disk-backed 复制时, RDB 文件生成完毕,多个 replicas 通过排队来同步 RDB 文件。

  • 当进行 diskless 复制时,master 节点会等待一段时间(下边的repl-diskless-sync-delay 配置)再传输以期望会有多个 replicas 连接进来,这样 master 节点就可以同时同步到多个 replicas 节点。如果超出了等待时间,则需要排队,等当前的 replica 处理完成之后在进行下一个 replica 的处理。

硬盘性能差,网络性能好的情况下 diskless 效果更佳

警告:无盘复制目前处于试验阶段

repl-diskless-sync-delay

当启用 diskless 复制后,可以通过此选项设置 master 节点创建子进程前等待的时间,即延迟启动数据传输,目的可以在第一个 replica 就绪后,等待更多的 replica 就绪。单位为秒,默认为5秒

repl-ping-replica-period

Replica 发送 PING 到 master 的间隔,默认值为 10 秒。

repl-timeout

默认值60秒,此选项用于设置以下情形的 timeout 判断:

  • 从 replica 节点的角度来看的 SYNC 过程中的 I/O 传输 —— 没有收到 master SYNC 传输的 rdb snapshot 数据;

  • 从 replica 节点的角度来看的 master 的 timeout(如 data,pings)—— replica 没有收到master发送的数据包或者ping;

  • 从 master 节点角度来看的 replica 的 timeout(如 REPLCONF ACK pings)—— master 没有收到 REPLCONF ACK 的确认信息;需要注意的是,此选项必须大于 repl-ping-replica-period,否则在 master 和 replica 之间存在低业务量的情况下会经常发生 timeout。

repl-disable-tcp-nodelay

master 和 replicas 节点的连接是否关掉 TCP_NODELAY 选项。

  • 如果选择“yes”,Redis 将使用更少的 TCP 数据包和更少的带宽向 replicas 发送数据。但这会增加数据在 replicas 端显示的延迟,对于使用默认配置的 Linux 内核,延迟可达40毫秒。

  • 如果选择“no”,则数据出现在 replicas 端的延迟将减少,但复制将使用更多带宽。

这个实际影响的是 TCP 层的选项,里面会用 setsockopt 设置,默认为 no,表示 TCP 层会禁用 Nagle 算法,尽快将数据发出, 设置为 yes 表示 TCP 层启用 Nagle 算法,数据累积到一定程度,或者经过一定时间 TCP 层才会将其发出。

默认情况下,我们会针对低延迟进行优化,但在流量非常高的情况下,或者当 master 和 replicas 距离多个 hops 时,将此选项改为“yes”可能会更好。

repl-backlog-size

设置复制的 backlog 缓冲大小,默认 1mb。backlog 是一个缓冲区,当 replica 断开一段时间连接时,它会累积 replica 数据,所以当 replica 想要再次重新连接时,一般不需要全量同步,只需要进行部分同步即可,只传递 replica 在断开连接时丢失的部分数据。

更大的 backlog 缓冲大小,意味着 replicas 断开重连后,依然可以进行续传的时间越长(支持断开更长时间)。

backlog 缓冲只有在至少一个 replica 节点连过来的时候 master 节点才需要创建。

repl-backlog-ttl

当 replicas 节点断开连接后,master 节点会在一段时间后释放 backlog 缓冲区。这个选项设置的是当最后一个 replica 断开链接后,master 需要等待多少秒再释放缓冲区。默认3600 秒,0表示永远不释放。

replicas 节点永远都不会释放这个缓冲区,因为它有可能再次连接到 master 节点, 然后尝试进行 “增量同步”。

replica-priority

replica-priority 是 Redis 通过 INFO 接口发布的整数,默认值为 100。当 master 节点无法正常工作后 Redis Sentinel 通过这个值来决定将哪个 replica 节点提升为 master 节点。这个数值越小表示越优先进行提升。如有三个 replica 节点其 priority 值分别为 10,100,25, Sentinel 会选择 priority 为 10 的节点进行提升。这个值为 0 表示 replica 节点永远不能被提升为 master 节点。

min-replicas-to-write

min-replicas-max-lag

//表示要求至少3个延迟<=10秒的副本存在
min-replicas-to-write 3  //下文中的 N
min-replicas-max-lag 10 //下文中的 M

从 Redis 2.8 开始,如果连接的 replica 延迟小于或等于M秒的个数少于N个(N个 replica 需要处于“online”状态),则 master 可能停止接受写入并回复 error。由于 Redis 使用异步复制,因此无法确保 replica 是否实际接收到给定的写命令,因此总会有一个数据丢失窗口。

原理如下:

  • replica 每秒钟都会 ping master,确认已处理的复制流的数量;

  • master 会记得上一次从每个 replica 都收到 ping 的时间,延迟就是根据 master 从 replica 接收的最后一次 ping 计算的;

  • 用户可以配置延迟不超过最大秒数的最小 replica 数;

此选项不保证 N 个副本将接受写入,但在没有足够的副本可用的情况下,将丢失写入的暴露窗口限制在指定的秒数内。

N 默认值为 0,M 默认值为10。任意一个设置为 0 表示不启用此功能。

replica-announce-ip 5.5.5.5

replica-announce-port 1234

Redis master 可以通过不同方式列出连接上来的 replicas 节点的地址和端口。如 Redis Sentinel 等会使用 “INFO replication” 命令来获取 replica 实例信息,master 的“ROLE“ 命令也会提供此信息。

这个信息一般来说是通过 replica 节点通过以下方式获取然后报告上来的:

  • IP:通过自动识别连接到 Socket 的信息自动获取

  • Port:一般来说这个值就是 replicas 节点用来接受客户端的连接的监听端口

但是,若启用了端口转发或者 NAT,可能需要其他地址和端口才能连接到 replicas 节点。这种情况下,需要设置这两个选项,这样 replicas 就会用这两个选项设置的值覆盖默认行为获取的值,然后报告给 master 节点。根据实际情况,你可以只设置其中某个选项,而不用两个选项都设置。

配置的介绍到这里就结束了,接下来我们把上边提到的概念串起来,聊一下主从复制的相关原理。

原理

系统的运行依靠三个主要的机制

  • 当一个 master 实例和一个 replica 实例连接正常时, master 会发送一连串的命令流来保持对 replica 的更新,以便于将自身数据集的改变复制给 replica ,包括客户端的写入、key 的过期或被逐出等等。

  • 当 master 和 replica 之间的连接断开之后,因为网络问题、或者是主从意识到连接超时, replica 重新连接上 master 并会尝试进行部分重同步。这意味着它会尝试只获取在断开连接期间内丢失的命令流。

  • 当无法进行部分重同步时, replica 会请求进行全量重同步。这会涉及到一个更复杂的过程,例如 master 需要创建所有数据的快照,将之发送给 replica ,之后在数据集更改时持续发送命令流到 replica 。

Redis 复制功能是如何工作的

每一个 Redis master 都有一个 replication ID :这是一个较大的伪随机字符串,标记了一个给定的数据集。每个 master 也持有一个偏移量,master 将自己产生的复制流发送给 replica 时,发送多少个字节的数据,自身的偏移量就会增加多少,目的是当有新的操作修改自己的数据集时,它可以以此更新 replica 的状态。

复制偏移量即使在没有一个 replica 连接到 master 时,也会自增,所以基本上每一对给定的 Replication ID, offset 都会标识一个 master 数据集的确切版本。

当 replica 连接到 master 时,它使用 PSYNC 命令来发送它记录的旧的 master replication ID 和它至今为止处理的偏移量。通过这种方式, master 能够仅发送 replica 所需的增量部分。但是如果 master 的缓冲区中没有足够的 backlog 或者 replica 引用了 master 不知道的历史记录(replication ID),则会转而进行一个全量重同步:在这种情况下, replica 会得到一个完整的数据集副本,从头开始。

说到这儿,那什么是全量同步,那什么又是增量同步呢?

全量同步

350ea867f8367ddceb5b03d79d8cce84.png

  1. replica 连接 master,发送 PSYNC 命令;

  2. master 执行 bgsave 开启一个后台保存进程,以便于生产一个 RDB 文件。同时它开始缓冲所有从客户端接收到的新的写入命令。

  3. 当后台保存完成时, master 将数据集文件传输给所有的 replica,并在发送期间继续记录被执行的写命令;

  4. replica 收到 RDB 文件之后,丢弃所有的旧数据,然后加载新文件到内存;

  5. replica 加载完成后,通知 master 发送所有缓冲的命令给 replica,这个过程以指令流的形式完成并且和 Redis 协议本身的格式相同;

  6. replica 开始接收命令请求,并执行来自 master 缓冲区的写命令。

注意:SYNC 是一个旧协议,在新的 Redis 中已经不再被使用,但是仍然向后兼容。因为它不允许部分重同步,所以现在 PSYNC 被用来替代 SYNC。

正常情况下,一个全量重同步要求在磁盘上创建一个 RDB 文件,然后将它从磁盘加载进内存,然后 replica 以此进行数据同步。如果磁盘性能很低的话,这对 master 是一个压力很大的操作。Redis 2.8.18 是第一个支持无磁盘复制的版本。在此设置中,子进程直接发送 RDB 文件给 replica 的 sockets 中,无需使用磁盘作为中间储存介质。

增量同步

master 把命令发送给所有的 replica 的同时,还会将命令写入 backlog 缓冲区里面。

当 replica 与 master 断开连接又重新连接之后,此时要判断 replica 的偏移量与 master 的偏移量的差集有没有超过 backlog 的大小,

  • 如果没有则给 replica 发送 CONTINUE,等待 master 将 backlog 中的数据发送给 replica;

  • 如果超过了则返回 FULLRESYNC runid offset,replica 将 runid 保存起来,并进行全量同步;

fe445e06846c14bf922221cc726024c5.gif

3671ffe7840c35ea48e3774467e5feb0.png

往期推荐

虚幻引擎5上的《黑客帝国》全新体验,爱了爱了

Medusa又一个开源的替代品

Log4j 第三次发布漏洞补丁,漏洞或将长存

5G专网,路在何方?

68064559b5dde713f7dc028bd337e35b.gif

点分享

ddeaf95515bbd62b5628a8f9e98c7f60.gif

点收藏

ec3e98dc21bb4524186dcbf9fbee5c49.gif

点点赞

cd752ab2e37547e39e7dc73729e7357e.gif

点在看

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

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

相关文章

基于英特尔® 优化分析包(OAP)的 Spark 性能优化方案

简介&#xff1a; Spark SQL 作为 Spark 用来处理结构化数据的一个基本模块&#xff0c;已经成为多数企业构建大数据应用的重要选择。但是&#xff0c;在大规模连接&#xff08;Join&#xff09;、聚合&#xff08;Aggregate&#xff09;等工作负载下&#xff0c;Spark 性能会面…

表格长度_知道你的成绩单是怎么打印的吗?超长Excel表格1页打印,拯救A4纸

中小学的成绩单&#xff0c;红色的一张榜真实的魔鬼&#xff01;每次都得瞄半小时才找得到自己的全部科目成绩&#xff0c;不知道是不是为了节省A4纸~到了大学我才知道A4纸的珍贵&#xff0c;字小算什么&#xff0c;打印论文恨不得双面打印。要是能八号字打印更好了~到了工作的…

苹果电脑上使用linux环境变量,mac系统下修改环境变量

苹果电脑使用率越来越高&#xff0c;在mac系统下研发&#xff0c;性能要比在windows下快不少&#xff0c;既然要开发&#xff0c;免不了要配置环境变量.下面是学习啦小编收集整理的mac系统下修改环境变量&#xff0c;希望对大家有帮助~~mac系统下修改环境变量的方法工具/原料os…

提升代码质量的方法:领域模型、设计原则、设计模式

简介&#xff1a; 我们可以列举出非常多质量差的代码的表现现象&#xff0c;其中最影响代码质量的两个表现是命名名不副实、逻辑可扩展性差&#xff0c;当一个新人阅读代码时&#xff0c;有时发现方法命名与实际逻辑对不上&#xff0c;这就让人感到非常疑惑&#xff0c;这种现象…

SphereEx 完成近千万美元 Pre-A 轮融资,连接企业数据与应用,构建新一代数据库生态引擎

2022年1月4日&#xff0c;创新型数据库基础软件提供商 SphereEx 宣布完成近千万美元 Pre-A 轮融资&#xff0c;本轮融资由嘉御资本领投&#xff0c;红杉中国种子基金、初心资本、指数创投跟投。指数资本担任独家财务顾问。这是继 2021 年 5 月份以来&#xff0c;SphereEx 完成的…

Javascript 机器学习的四个层次

简介&#xff1a; Atwood定律说&#xff0c;凡是可以用Javascript实现的应用&#xff0c;最终都会用Javascript实现掉。作为最热门的机器学习领域&#xff0c;服务端是Python的主场&#xff0c;但是到了手机端呢&#xff1f;Android和iOS里默认都没有Python。但是有浏览器的地方…

运维监控再添新品,F5联合智维数据推出应用质量主动拨测解决方案

一直以来&#xff0c;业务的稳定性和客户体验是企业各个部门都关注的重点&#xff0c;也是企业数字化转型的重要支撑。但在实际的运维环境中&#xff0c;如果只从服务端的视角进行监控&#xff0c;往往会出现数据中心内部监控体系显示设备状态正常&#xff0c;但却收到了客户端…

排查指南 | 两个案例学会从埋点排查 iOS 离线包

简介&#xff1a; 首次打开离线包白屏以及报错“-1009”等该如何处理呢&#xff1f; 离线包原理 以一次启动离线包的流程为例&#xff0c;离线包的加载流程分为两种场景&#xff0c;第一种是离线包下载好的场景&#xff0c;流程如图1所示&#xff0c;第二种是离线包没下载好的…

如何设计可靠的灰度方案

简介&#xff1a; 一个较大的业务或系统改动&#xff0c;往往会影响整个产品的用户体验或操作流程。为了控制影响面&#xff0c;可以选取一批特定用户、流程、单据等&#xff0c;只允许这一部分用户或数据按照变更后的新逻辑在系统中流转&#xff0c;而另一部分用户仍然执行变更…

linux如何设置mac快捷键,在Ubuntu上使用macOS的快捷键

因为常用机一台Mac&#xff0c;一台Linux&#xff0c;都频繁使用&#xff0c;两个系统不同的快捷键已经让人精神分裂了&#xff01;macOS几乎所有的快捷键都基于command键&#xff0c;全选(cmda)、复制(cmdc)、粘贴(cmdv)、开关标签页(cmdt/w)、切换窗口(cmdtab)、保存(cmds)、…

IDC报告:阿里云领跑中国数据库市场年度份额首超传统厂商

简介&#xff1a; IDC报告显示&#xff0c;2020年中国关系型数据库软件市场规模达到121.8亿元&#xff0c;同比增长36.5%。其中&#xff0c;以公有云模式部署的关系型数据库市场占比达到51.5%&#xff0c;首次超过传统线下部署模式市场规模&#xff0c;预计到2025年将继续走高&…

稳若磐石的「云上奥运」背后,是云计算新界面的崛起

简介&#xff1a; 今年奥运会首次采用阿里云支撑全球转播&#xff0c;这是奥运迈入数字时代的重要一步。对于这届特殊的奥运会来说&#xff0c;技术至关重要。我们相信&#xff0c;这次创造历史的实践&#xff0c;将为未来支撑更多体育爱好者把“云观赛”转变为参与国际体育赛事…

10分钟!构建支持10万/秒请求的大型网站

简介&#xff1a; 应用网关作为应用的统一接入层&#xff0c;它的发展和演进也是伴随着应用架构的变化&#xff0c;大家都知道企业应用从最早期 SOA 时代发展到微服务的时代。在 SOA 时代&#xff0c;传统的企业服务总线承担了企业应用的统一接入层&#xff1b;但是发展到微服务…

服务实体经济、战略级行业再下一城,钉钉发布制造行业解决方案2.0

编辑 | 宋慧 出品 | CSDN 云计算 1月5日&#xff0c;钉钉召开主题为“数字新生”的2022制造业钉峰会。会上&#xff0c;钉钉正式发布制造行业解决方案2.0&#xff0c;该方案以“码上制造”产品为制造行业专属底座&#xff0c;提供设备上钉、计件日结等基础产品&#xff0c;同时…

阿里云边缘云全新架构升级,助力CDN操控新体验

简介&#xff1a; 本次升级根据上万企业客户的使用反馈和行业应用特征&#xff0c;从简单开通到个性化定制&#xff0c;从内容分发到边缘计算完整解决方案&#xff0c;对客户侧的使用体验进行了全局梳理和全链路优化&#xff0c;推进边缘云CDN操控革新&#xff0c;并逐步构建面…

key redis 遍历_快乐运维Redis大数据量查询与清理

一、 keys命令(生产环境禁止使用)简单粗暴&#xff0c;由于Redis单线程这一特性&#xff0c;keys命令是以阻塞的方式执行的&#xff0c;keys是以遍历的方式实现的复杂度是 O(n)&#xff0c;Redis库中的key越多&#xff0c;查找实现代价越大&#xff0c;产生的阻塞时间越长。key…

linux部分基础命令总结,Linux常用基础命令总结

近期自己学习了一下Linux&#xff0c;写这篇博客以便于对自己的一个总结&#xff0c;记录自己的学习情况&#xff0c;奥利给&#xff01;想对Linux熟练掌握&#xff0c;就必须学会它的操作命令&#xff0c;虽然可能会花费一些时间&#xff0c;不过从长远的角度来说&#xff0c;…

MaxCompute执行引擎核心技术DAG揭秘

简介&#xff1a; 作为业界少有的EB级数据分布式平台&#xff0c;MaxCompute每天支撑上千万个分布式作业的运行。这些作业特点各异&#xff0c;既有包含数十万计算节点的超大型作业&#xff0c;也有中小规模的分布式作业。不同用户对于不同规模/特点的作业&#xff0c;在运行时…

自定义导航页_带你回归“真”的导航页

曾经浏览器主页是我们流向各大网站的要道从这里我们可以轻而易举的去往想要去的网站特别是在信息和专业性越来聚集的今天传统的主页难免令人有些审美疲劳或者说花里胡哨hao1232345是这些老牌的主页网站有些审美疲劳随着导航越来越细节化广告也越来越多美感也更少360主页qq主页百…

Redis 压缩列表原理与应用分析

作者 | 西瓜来源 | JAVA架构进阶之路摘要Redis是一款著名的key-value内存数据库软件&#xff0c;同时也是一款卓越的数据结构服务软件。它支持字符串、列表、哈希表、集合、有序集合五种数据结构类型&#xff0c;同时每种数据结构类型针对不同的应用场景又支持不同的编码方式。…