大家好,我是悟空。
玩过王者荣耀的同学,应该都知道里面有个英雄叫做镜
,她释放技能时,会出现一个长相一模一样的分身,而且动作也是一样的。
那么我们今天要讨论的主从架构原理其实就是多个节点中有一个作为本体,其他节点作为分身存在,但是本体和分身的数据都是一样的。
为了保证缓存的高可用,我们经常听到采用主从架构来保证高可用,那如何去理解主从架构核心原理呢?
这次我们还是用最熟悉的 Redis 缓存来理解主从架构,只要理解了一个主从架构,其他技术的主从架构都是一通百通。
Redis 的主从架构,其实就是利用多副本,将一份数据同时保存在多个实例上。单个实例出现故障后,一般都会过一段时间才能恢复,那么其他节点还是可以提供服务的。
本篇我会带着大家一起探讨缓存的主从架构几个问题:
Redis 单机我们都玩得很溜了,那单点架构会带来什么问题?
服务的快速转移和恢复数据,其实是高可用的范畴,我们就可以通过主从架构来做到了。
文中最开始也提到过这么一段话,划重点:
Redis 的主从架构,其实就是利用多副本,将一份数据同时保存在多个实例上。单个实例出现故障后,一般都会过一段时间才能恢复,那么其他节点还是可以提供服务的。
Redis 主从架构其实就是主从库模式,而主从库的模式可以分为三种拓扑结构:一主一从结构、一主多从结构、树状主从结构。而如何去构建这种结构其实也很简单,就是配置下多个节点上 Redis 的配置文件就可以了。如下所示:
slaveof
当然还有一些其他参数配置,就不在本篇讲解,下篇主从的部署教程其实已经写好了,后续发出来。
下面我们接着来看这几种主从架构:
而对于一主多从结构,还可以再扩展一点:当日常开发中需要执行一些比较耗时的读命令时,比如 keys
、sort
等,可以用其中一个从节点专门作为耗时查询用的从节点,避免慢查询对主节点造成阻塞,而影响服务的稳定性。我们也可以用图来进行说明:
从节点复制主节点的数据后,就相当于给主从节点备份了,所谓的有备无患就是这个意思。那么主从复制的原理是怎么样的?其实主要就是三种复制方式:持续复制、全量复制、部分复制。
当有客户端的写命令请求到主节点后,主节点会做两件事:命令传播
和将写命令写入到复制积压缓冲区
。
原理图如下:
用于主从节点第一次复制的场景。这在我们的软件开发中也很常见,比如你要把第三方的用户数据同步到自己的系统中,一开始肯定是把存量用户一次性给复制过来,后续有新增或更新的用户就采用增量更新就可以了。
当然全量复制的时候,数据量很大时,就会对主从节点和网络造成很大的开销,也就是常说的复制风暴
,所以要避免不必要的全量复制,这个后面再讲怎么避免。
我们先来看下全量复制的原理图,然后我再来详细解释每一步怎么做的。
全量复制总共可以分为 9 步:
(1)从节点发送 psync 命令进行数据同步,会发送 psync 命令,来告诉主节点我想干啥。
psync 命令格式如下:
psync {runId} {offset}
runId 是 每个 Redis 实例启动时随机生成的一个 ID,用来唯一标记这台 Redis 实例。由于第一次复制时,刚启动时会随机生成 runId,只有自己知道,外人是不知道的,所以从节点是不知道主节点的 runId 的,这个时候发送 psync 命令时 runId 就是一个问号。第一次复制时,offset 默认为 -1。
(2)主节点响应从节点,要开始全量复制了哦。
主节点响应从节点时分三种情况:全量复制、部分复制、旧版全量复制流程,这三种的区别后面会专门讲到。
而这个阶段就是全量复制:主节点告诉从节点,要开始全量复制了哦。响应如下命令:
FULLRESYNC
runId
就是主节点 id,offset
为复制偏移量。
(3)从节点收到主节点响应的 runId 和 offset,将其保存到从节点本地,这两个参数以后会用到。
(4)主节点在后台执行 bgsave
命令保存 RDB 文件到主节点的本地。
(5)主节点将第四步生成的 RDB 文件发送给从节点,子节点收到 RDB 文件后,保存到本地,后面会用到。这个发送的过程也可能直接超时。比如一个 6 GB 的 RDB 文件,100 MB 带宽下,至少需要 60 秒的传输时间,很容易超出默认配置的超时时间。那么从节点将放弃接收 RDB 文件,并清理已经下载的临时文件,导致全量复制失败。所以推荐不要超过 6 GB,如果 RDB 文件实在太大了,可以调大 repl-timeout 超时参数。
(6)在第五步的时候,主节点也没有闲着,会往另外一个缓冲区写东西,就是来自客户端的写命令数据。这个缓冲区叫做:复制客户端缓冲区。等第五步完成后,主节点就把这个缓冲区的数据发送给从节点。注意:对于高流量写入的场景,很容易就把复制客户端缓冲区给占满了,如果 60 秒内缓冲区消耗持续大于 64 MB 或者直接超过 256 MB 时,主节点将直接关闭复制客户端连接,造成全量不同失败。
(7)从节点在第五步保存完 RDB 文件后,就会把自身的旧数据清空。
(8)历经磨难,从节点终于可以开始加载 RDB 文件了,但是对于较大的 RDB 文件,加载 RDB 文件,进行数据恢复,还是非常耗时的,如果从节点负责响应读命令,则可能拿到过期或错误的数据。
(9)从节点加载完 RDB 后,如果当前节点开启了 AOF 持久化功能,从节点会执行 bgrewriteof 操作,保证 AOF 持久化文件可以立刻使用。
总结下全量复制的步骤:
从节点给主节点发送命令;
主节点回复从节点,要开始全量复制了;
从节点保存主节点信息;
主节点开始生成 RDB 快照文件;
RDB 文件发给从节点,主节点发送 RDB 文件;
主节点发送缓存的客户端命令;
从节点清空旧数据;
从节点加载 RDB 文件;
从节点执行 AOF 操作。
由上面的几个步骤可以看出,全量复制是非常耗时的,可能比较大的时间开销如下:
所以除了第一次需要采用全量复制外,其他场景应该避免全量复制的发生。下面介绍另外一种复制方式,可以极大提高复制的效率。
这个可以理解为增量更新,比如和第三方系统对接时,如果第三方有数据更新,定期进行增量更新就可以了。
而 Redis 主从的部分复制就是指当主从之间的网络故障等原因造成持续复制中断了,当从节点再次连上主节点后,主节点就补发数据给从节点,避免了全量复制的过高开销。补发数据的来源就是复制积压缓冲的数据。
原理图如下所示:
部分复制总共分为六步:
(1)当主节点之间失联后,如果时间超过了 repl-timeout 时间,主节点就认为从节点发生故障了,中断连接。
(2)主节点其实一直都在把客户端写命令放入复制积压缓冲区,所以即使断连了,主节点还是会保留断连期间的命令,但因为队列是固定的,当写命令太多时,就会导致部分命令被覆盖了。
(3)主从节点恢复连接。
(4)从节点发送 psync 命令给主节点,带有 runId 和 offset 参数,runId 是上一次复制时保存的主节点的 runId值,offset 是从节点的复制偏移量。
(5)主节点接收到从节点的命令后,先判断传过来的 runId 是否和自己匹配,如果不匹配,则进行全量复制;如果 runId 匹配,则响应 CONTINUE,告诉从节点,可以进行部分复制了。我要把复制积压缓冲区的数据发给你了哦,请准备好接收。
(6)主节点根据子节点发送的偏移量,将复制积压缓冲区的数据发送给子节点。
那复制积压缓冲区到底是怎么来根据偏移量来计算要发送哪些缓存数据的呢?我们接着往下看。
复制积压缓冲区有几个特点:
从节点重新连上主节点后,会发送 psync 命令,携带着偏移量 offset。比如 offset = 125,然后主节点拿着这个 125 去复制积压缓冲区找,125 正好在里面,然后就会执行部分复制的操作,将 125 以后的缓冲数据发送给从节点。
如果 offset =10,主节点拿着这个 10 去复制积压缓冲区找,发现队列中最早的 offset 是 100,所以 100 之前的字节都被覆盖了,那么子节点就不能通过复制积压缓冲区拿到完整数据,所以只能通过全量复制的方式来同步。这个时候主节点就会发送一个 +FULLRESYNC 的命令给子节点,告诉子节点,兄弟,你来得太晚了,只能使用全量同步的方式了。
所以为了合理设置复制积压缓冲区的大小,有个计算公式推荐给大家:
2 * 恢复连接的时间(s) * 主节点写缓冲区的速度(MB/s)。
比如从节点需要 10 s 才能连上主节点,而主节点在这期间每秒产生 5 MB 的写数据,那么复制积压缓冲区的大小可以设置为 100 MB。(2 * 10 s * 5 MB/s = 100 MB)
在上面提到从节点不管是全量复制还是部分复制,最开始都会发送一个 psync 命令给主节点,那么主节点会根据这个命令携带的参数 runId 和 offset,来决定如何响应。
原理图如下所示:
说完上面主从节点的连接的结构,接下来的问题是这些节点如何在复制时保持连接呢?
也就是说主节点和从节点如何知道对方还存活着?其实就是通过心跳检测,这个在服务注册和服务发现里面经常提到。而主节点和从节点都会向对方发送心跳检测的命令。
主节点发送命令:
主节点会每隔 10 秒对从节点发送 ping 命令(也就是 10 秒一次心跳),从节点接收到 ping 命令后,会进行响应,所以这个 ping 命令可以用来判断从节点的存活性和连接状态。10 秒检测一次是可以调整的,用参数 repl-ping-slave-period 控制发送频率。
从节点发送命令:
而从节点也会每隔 1 秒发送一个命令给主节点。这个命令还挺拗口,叫做 REPLCONF ACK {offset} ,作用就是给主节点上报自身当前的复制偏移量
的,这个偏移量用来检查复制数据是否丢失,如果丢失,主节点将补发丢失的数据。
补发丢失数据的操作和部分复制操作的区别:
补发丢失数据操作在主从服务器没有断线的情况下执行。
而部分复制操作是在主从服务器断线后重连之后执行。
从节点发送这个命令还可以实时检测主从节点的网络状态,类似于我们在命令行窗口 ping 服务器 ip,如果 ping 不通,则表示双方的网络连接有问题。
另外从节点的命令还可以用来计算主从的通信延迟,正常的延迟在 0~1 秒之间,如果超过了默认的 60 秒,则判定从节点下线,断开复制,等重新上线后,复制继续。而断开期间丢失的数据,则可能需要全量复制或部分复制,取决于从节点的 offset 偏移量在不在复制积压缓冲区。
运维人员肯定是要关注,然后就是架构师和技术专家,他们对这些技术是需要有深入了解的。当然掌握了主从复制原理,其他技术栈的复制原理也是类似,甚至在业务场景下,和第三方同步数据也可以用到里面的思想。
其实 Redis 有更好的高可用架构方案:集群部署。
而集群部署就是将多个主从结构进行横向扩容 + 分片存储 + 哨兵。这里面的几个名词放到后面的文章再讲,本篇主要针对主从架构。
所以真实的大型电商的生产环境一般都是采取集群部署的方式,一些规模较小的系统直接单机就可以搞定,再提高点可用性的话,就用主从架构,实现读写分离。其实架构就是综合考虑学习成本+部署成本+维护成本得到的一种最合适的方案。一味追求高可用,不结合实际的话,用处不大,这里推荐大家看下《架构整洁之道》这本书。
原因:
因为主从架构只有一个主节点,所以主节点宕机后,影响整个系统运行。
如何监控主节点?用 Redis 哨兵机制。
如何增加主节点?用 Redis 集群。
主从设置为读写分离后,需要客户端绑定主节点进行读写,绑定从节点进行读,这个是比较麻烦的事情。
主从读写架构就不能自行判断往哪个服务器读,哪个服务器写吗?其实需要我们自己来控制。
主从模式肯定比单机复杂,维护成本也较高。
本篇从宏观和微观上讲解了主从架构的原理:
微观视角:涉及了主从架构的三种拓扑结构来应对不同的场景,然后深入讲解了持续复制、全量复制、部分复制的区别。以及大家特别关心的复制积压缓冲区,客户端缓冲区的使用场景,以及会遇到的坑。
宏观视角:讲解了主从架构谁需要来关心、什么时候用、以及主从架构会带来的问题。
后续:主从架构的部署和压测下篇安排~