|
# B8 f: F% O0 h8 H8 T# W<p>上一节提到了 Redis 的持久性,也就是在服务器实例宕机或故障时,拥有再恢复的能力。但是在这个服务器实例宕机恢复期间,是无法接受新的数据请求。对于整体服务而言这是无法容忍的,因此我们可以使用多个服务器实例,在一个实例宕机中断时,另外的服务器实例可以继续对外提供服务,从而不中断业务。Redis 是如何做的呢?Redis 做法是<strong>增加冗余副本</strong>,<strong>将一份数据同时保存在多个实例</strong>上。那么如何保存各个实例之间的数据一致性呢?Redis 采用<strong>主从库读写分离模式</strong>来保证数据副本的一致性。</p>
7 Q, W% c! o5 Q1 o! K: n<h2 id="一主从复制介绍">一、主从复制介绍</h2>
5 E+ n4 W4 |% D7 z' ~<p>在 Redis 中提供复制的服务器称为主服务器(master),被主服务器进行复制的服务器称为从服务器(slave )。但是在复制方式上有所区别,主从库之间采用的是读写分离的方式:</p>
7 @# S3 @4 o" c) }0 w& M7 O<ul>! G7 I3 I2 I: c0 _" q N- | T7 {
<li><strong>读操作</strong>:主库和从库都可接收</li>% r5 k7 J3 ~. Z2 v6 c
<li><strong>写操作</strong>:写操作先在主库中执行,然后再将操作同步给从库</li># G; T! m ^$ l) I8 N ~ j' E
</ul>
$ J0 S/ j* g" P5 q<p><img src="https://img2022.cnblogs.com/blog/1707576/202202/1707576-20220211213252722-390582213.jpg" ></p>
1 D2 v S8 m" J) h7 Z<p>为何要采用读写分离,因为如果在不同实例上执行修改操作,要保证实例之间的一致性就必须加锁、实例间的协商等操作,会带来巨额的开销。如果采用读写分离,数据的修改迁移到主库上进行,然后再同步到从库上,就可以达到不使用锁达到数据一致性的效果。</p>3 H& R! Y6 `( A7 l' Q
<h2 id="二主从复制原理">二、主从复制原理</h2>
. ?! o% k' Y" E9 E! B" P" P. h<p>主从复制是如何进行复制的,是一次性全部复制,还是分批一批一批的复制?而且如果复制中网络中断,数据还能保持一致性吗,其内部原理是怎样的?Redis 的复制功能主要有两个操作:</p>
# f+ X# q( _0 L6 S- K3 ?<ul>
" d8 o: K' \/ x6 r<li>$ ~; s% e! i, s1 Z5 r+ H
<p><strong>同步(sync)</strong>:同步操作是将从库的数据状态更新至主库当前所处的状态,主要有全量复制和增量复制两种</p>
6 ~* W6 _( P6 h2 m</li>9 ^# M, S" |- h, R. C
<li>. E. r" T7 k& h" M" k
<p><strong>命令传播(command propagate)</strong>:命令传播是作用在主库的数据库状态被修改后,主从库之间的数据库状态出现不一致,让主从服务器的数据库状态重新回到一致性。</p>2 s V4 V' r3 r
</li>
9 h- o( m2 f3 I# U</ul>
0 e/ p; e* S5 u( o+ M+ j8 [/ H<h3 id="21-同步操作">2.1 同步操作</h3>
/ o; C0 g. x3 T$ ^3 \4 s& @<h4 id="211-全量复制">2.1.1 全量复制</h4>" g0 T' T9 m) ]
<p>在同步操作中,我们首先来看看全量复制,顾名思义也就是主库将所有的数据传送给从库。因为主从库初始化需要传输全部数据,所以第一次同步其实就是一次全量复制。当启动多个 Redis 实例时,主从库间就可以通过 <code>replicaof</code>(Redis 5.0 前使用的是 <code>slaveof</code>)命令形成主从库的关系,在这期间从库就会全量复制当前主库的数据库状态。主要分为三个阶段:</p>& `5 x, Q% ]1 X7 a
<ol>, _; ~ h4 r$ t( b4 e' Y9 a, }4 m [
<li>! e" J( Z: d& ~) ? b E
<p><strong>主从库建立连接</strong>:主要是为全量复制做准备,在这一步,从库和主库建立起连接,告知主库进行数据同步,待主库确认回复后,主从库之间可以开始同步。主要的操作是从库给主库发送 <code>psync</code> 命令,主库根据这个命令参数来启动复制。而 <code>psync</code> 命令包含了主库的 <code>runID</code> 和 复制进度 <code>offset</code>两个参数:</p>
6 ^6 h3 s9 C3 y8 ?& |<ul>
- } a/ R" S1 F$ Z: v0 D2 R<li><strong><code>runID</code></strong>: 是每个 Redis 实例启动时随机生成的 ID,它是用来唯一标记这个实例。而当主从库第一次复制时,从库不知道主库的 <code>runID</code> 。因此将 <code>runID</code> 设置为 “?”。</li>9 y5 O3 _; f O( ] F; S7 c
<li><strong><code>offset</code></strong>:第一次设置为 -1,表示第一次复制</li>0 i3 j/ A( `! M9 L
</ul>
! F" e) t2 ?5 _7 N( W! [<p>主库在接收到 psync 命令后,会利用 FULLRESYNC 响应命令带上两个参数:主库 <code>runID</code>和主库目前的复制进度 <code>offset</code> 来返回给从库。而从库在收到响应后会记录下主库传递的这两个参数。</p>
) k6 W- X: i& A* @% o8 E</li>3 c, a9 ?# b8 k* }7 y
<li>
. Y7 a1 z6 q( q) h- }* d. D% c( R2 c<p><strong>主库将所有数据同步给从库</strong>:数据同步给从库后,从库接收到数据后,在本地完成数据加载。这个过程主要依赖于内存快照生成的 RDB 文件。</p>
9 s& K9 [& ^; @; v% A7 L<p>内部具体的流程是,主库先执行 <code>bgsave</code> 命令,执行持久化生成 RDB 文件,并将该文件发送给从库。从库在接收到 RDB 文件后,会先清空当前数据库,然后再加载 RDB 文件。</p>
+ x( L$ q$ ^/ L! G. b9 P+ p<p>在主库同步从库过程中,我们知道 <code>bgsave</code>命令系统会 fork 一个子线程来创建 RDB 文件,另外的主线程可以继续处理命令请求。而在同步过程中的新增的写操作,主库会在内存中用专门的 <code>replication buffer</code> 来记录这些写操作。</p>
" G' x2 w+ e& d5 x2 |! \; a/ C</li> f4 U' S- F- H7 N1 t
<li>& q4 c. z! h. g! v
<p><strong>主库将新接收的写操作再发送给从库</strong>:当主库完成 RDB 文件发送后,就会将 <code>replication buffer</code> 中的写操作发送给从库,从库再执行这些写操作。到此完成所有的操作数据同步。</p>
- k8 E& ? t- x* G8 E</li>% |, S, v3 O# w, h
</ol>
5 T3 O) q+ s! z( Z2 V<p>结合一个实例来展示三个阶段:</p>5 h6 m- ?6 @+ U& y& w2 h7 w2 u
<p>现有两个实例,实例1(ip: 172.16.19.3)和实例 2(ip: 172.16.19.5)。在从实例2上执行以下命令后,实例2 就变成了实例 1 的从实例。执行以下命令后,实例2 就变成了实例1 的从库,并开始复制数据:</p>) ^# Z0 y; d7 a. [
<pre><code class="language-sh">replicaof 172.16.19.3 6379 I( @2 R0 H: u C( m$ S9 B2 g
</code></pre>
9 e* p2 b/ C& i% F6 D1 G# v8 C<p><img src="https://img2022.cnblogs.com/blog/1707576/202202/1707576-20220211213311291-1707870632.jpg" ></p>
+ q( ^* b/ \9 ^+ @<h4 id="212-主从从模式复制">2.1.2 主从从模式复制</h4>
& d) d5 ?. \9 o3 W$ H# n<p>在一次全量复制中,对于主库而言需要完成两个消耗资源的操作:发送 RDB 文件和 <code>repl buffer</code> 文件。所以如果请求的从库数量较多时,虽然主库会 fork 子线程进行生成 RDB 文件,但是从库请求数量过多,也会导致主线程 fork 操作过多,最终也会阻塞主线程的其他正常请求。所以,为了解决这个问题,我们可以<strong>通过 “主-从-从”模式来将主库生成 RDB 和传输 RDB 的压力,以级联的方式分散到从库上</strong>。如下图:</p>
$ y5 G0 O7 z. L% T<p><img src="https://img2022.cnblogs.com/blog/1707576/202202/1707576-20220211213330954-674647319.jpg" ></p>
+ t0 n* o; ^$ t% [# h( u' f! O- j<p>具体操作是: 在部署主从集群的时候,手动选择一个从库(比如选择内存资源配置较高的从库)用于级联其他的从库。然后可以再选择一些从库(如三分之一的从库),在这些从库上执行下面命令,让它们和刚才所选的从库建立起主从关系:</p>( ]6 n) A2 H- W8 [: t4 A
<pre><code class="language-sh">replicaof 所选从库的IP 6379' X) `) }; D) J
</code></pre>
3 W2 M8 S$ I B0 {5 K<p>这样,这些级联的从库不用和主库进行交互,而只需要和连接的从库进行写操作同步即可。这样就可以减轻主库的压力了。</p>6 j) R' z( a4 Q; _( }
<h4 id="213-增量复制">2.1.3 增量复制</h4>
" I J! T% n( I0 Z j<p>在主从库完成第一次的全量复制后,它们会形成一个网络连接,主库会通过这个连接将后续收到的命令操作再同步给从库。这个过程也称作为<strong>基于长连接的命令传播</strong>。这个长连接可以避免频繁的建立连接开销,后续我们会再提命令传播。</p>
# V. D) m! q2 F<p>那么为什么需要增量复制呢,因为连接过程中存在着<strong>网络连接和阻塞</strong>,如果网络连接中断,主从库之间就无法实现命令传播。那如果再次进行全量复制,其开销就有点得不偿失。所以新设计出了增量复制,而与全量复制不相同,增量复制只会把主从库网络断联期间主库收到的命令同步给从库。其中重点就是利用<code>repl_backlog_buffer</code> 缓冲区,上面我们知道,在全量复制时,主库会把写操作命令写入 <code>replication buffer</code> ,与此同时,也会把这些操作命令写入 <code>repl_backlog_buffer</code> 缓冲区中(<code>repl_backlog_buffer</code> 是一个环形缓冲区,主库会记录自己写到的位置,从库则会记录自己已经读到的位置)。如下图:</p>$ N% {+ T. m1 q$ g- L% G
<p><img src="https://img2022.cnblogs.com/blog/1707576/202202/1707576-20220211213421342-676422844.jpg" ></p>: s( M5 y4 y) C6 h" U0 o
<p>主库在缓冲区的写位置偏移量就是 <code>master_repl_offset</code>,从库的读位置偏移量是 <code>slave_repl_offset</code>。正常情况下两个偏移量基本相等。</p>, {) Q m% x' \ u
<p>接下来我们看看网络连接断开时,主库有可能会收到新的写操作命令,一般而言,<code>master_repl_offset</code>会大于 <code>slave_repl_offset</code>。所以当主从库网络连接后,主库只需要将 <code>master_repl_offset</code> 和 <code>slave_repl_offset</code> 中间的命令操作同步给从库。</p>
! |0 p( {: @/ c: K0 I( D* D8 B<p>如上图中的 <code>repl_backlog_buffer</code>示意图,主库和从库之间相差了<code>put d e</code>和 <code>put d f</code> 两个操作,在增量复制时,主库只需要将它们同步给从库即可。</p>2 r0 y+ u/ U/ t
<p>我们再来看一下增量复制的流程:</p>$ j3 d( r9 l# b! h2 E Q1 O
<p><img src="https://img2022.cnblogs.com/blog/1707576/202202/1707576-20220211213439896-844044503.jpg" ></p>
! L- A0 C3 Z# v: [0 ~& p' C' Q<p><code>repl_backlog_buffer</code> 是一个环形缓冲区,所以存在这样的情况:在缓冲区写满后,主库会继续写入,这样就会覆盖之前的写操作,那么这有可能就会导致主从库之间的数据不一致。此时,我们可以调整 <code>repl_backlog_size</code> 这个参数,实际情况下可以根据应用调整 <code>repl_backlog_size</code> 的大小。</p>- r. @8 A0 P! E' o/ C/ K
<h3 id="22-命令传播">2.2 命令传播</h3>
( S8 C+ Q& N7 Z/ V+ a3 g1 d1 }) q<p>在上述初始化全量复制结束后,主从库两者的数据库状态达到一致完成了同步,后面两者则处于长连接状态。此时主库只需要一直将自己执行的写命令发送给从库,从库一直接收并执行主库发来的写命令即可保证主从库的数据一致性了。这个时候主从库会互相成为对方的客户端。</p>7 j) C, X; Z8 o. F% d
<h4 id="221-主库的命令传播">2.2.1 主库的命令传播</h4>
9 z% t8 I2 T% w- r$ b<p>当完成了同步之后,主从库就会进入命令传播阶段,这时主库只要一直将自己执行的写命令发送给从库,而从库只要一 直接收并执行主库发来的写命令,就可以保证主从库保持数据操作一致性。</p>
% ~. u7 W5 v/ h% d1 V$ i<h4 id="222-从库的心跳检测">2.2.2 从库的心跳检测</h4>
5 u. z" D: z" k$ E: C0 d<p>在该阶段,从库默认会以每秒一次的频率向主库发送命令:</p>
0 Q* a D& `5 Q' v. H<pre><code class="language-sh">REPLCONF ACK <replication_offset>
) Y/ W( N; \" j1 G</code></pre>
' `, I& m! A4 \' K# d<ul>
" p7 w. I' t- Z; C+ {+ H<li>
- e. o+ \! z0 V<p>replication_offset 是从库当前的复制偏移量</p>
{, ~1 \* ~5 ~1 }</li>
, I2 p9 }8 t$ h9 l2 ]<li>3 d1 A0 l2 \' r% u% e- H# \
<p>REPLCONF ACK 命令的作用有</p>
0 x5 L; m* v0 r$ `; q9 r4 s+ Y<ul>
' |) F. d4 ]+ ~- ?0 z& k<li># X) j& o, m$ }9 }+ v" U/ E5 Y8 h
<p>检测主从库的网络连接状态:当主库超过一秒钟未收到从库发来的 REPLCONF ACK 命令,那么主库就会知道主从库之间连接出现问题。</p>, e% { r) p; S; K& d5 @7 g
</li>* D$ t6 D% t* y. _3 G
<li>
2 J, V; z2 u% N) s' I8 z1 `/ h<p>辅助实现 min-slaves 选项:Redis的<code>min-slaves-to-write</code>和<code>min-slaves-max-lag</code>两个选项可以防止主库在不安全的情况下执行写命令。</p>6 k( Q9 b9 g) y+ c- Z4 l* a6 n) z
<ul>
+ c: o5 H9 o) j- d+ I<li>6 p. Y2 f1 X/ ~5 [* A' T
<p>举个例子,如果我们向主库提供以下设置:</p>
" f7 s3 f8 ?- U8 L<pre><code class="language-sh">min-slaves-to-write 3
( S$ G8 W3 L' T# t9 y- Mmin-slaves-max-lag 10( k! J$ d- F7 c% U/ y1 R
</code></pre>
S% ^- E% [2 U7 b' u<p>那么在从库的数量少于3个,或者三个从库的延迟(lag) 值都大于或等于10秒时,主库将拒绝执行写命令,这里的延迟值就是上面提到的INFO replication命令的lag值。</p>3 e8 W, {0 E9 o9 H8 W
</li>
1 N! X$ \; }2 k8 {! A- L</ul>7 q6 s2 ~+ d8 H) O( j
</li>
2 c' z& l0 d5 h( @& F<li>) h" m# U& m* e0 j
<p>检测命令丢失: 如果因为网络故障,主库传播给从库的写命令在半路丢 失,那么<strong>当从库向主库发送REPLCONF ACK命令时,主库将发觉从库当前的复制偏移量少于自己的复制偏移量</strong>,然后主库就会根据从库提交的复制偏移量,在复制积压缓冲区(<code>repl_backlog</code>)里面找到从库缺少的数据,并将这些数据重新发送给从库。</p>( J! c |+ v& x$ _ P
</li>
& s, e% F1 _' }, x8 h7 w h</ul>( ]9 f9 x2 s J: h
</li>
, O4 p1 B" o% v' T7 h) M/ b: c</ul>7 D& a3 W L6 |: I5 A1 T3 V+ m
<p>总结就是在传播命令阶段,主库通过向从库传播命令来更新从库的状态,保持主从库一致。而从库则通过向主库发送命令来进行心跳检测,以及命令丢失检测。</p># }6 T. W6 \" P. ^. Q
<h2 id="三主从复制的面试题">三、主从复制的面试题</h2>
) d' R- {1 N# P0 B<h3 id="31-aof-记录的操作命令更全相比-rdb-丢失的数据更少那么为什么主从库间的复制不使用-aof-">3.1 AOF 记录的操作命令更全,相比 RDB 丢失的数据更少,那么为什么主从库间的复制不使用 AOF ?</h3>3 j! \0 i+ G$ d
<ul>
7 ^" y4 M& x, Z& p C<li><strong>RDB 文件是以压缩二进制的方式存储,文件小</strong>,所以在从库加载 RDB 文件时,速度会很快。而 AOF 需要依次重放每个写命令,过程相对 RDB 文件的方式要慢的多。</li>
% O- D% m; `# u9 ~; q<li>如果使用AOF 做全量复制,打开 AOF 功能需要选择文件刷盘的方式,选择不当会严重影响 Redis 的性能。而 RDB 持久化方式只需要定时备份和主从全量复制数据时才会触发生成一次快照。在大多数丢失数据不敏感的业务场景,其实是不需要开启 AOF 的。</li>
+ f9 U' A, r1 W5 ^6 w9 c</ul>5 s4 [, o$ T! P9 C; U8 J) T5 t
<h3 id="32-如何处理读写分离中的问题">3.2 如何处理读写分离中的问题</h3>
$ [8 i f7 o! c2 \<p>Redis 的读写分离可以实现 Redis 的读负载均衡,能够提高 Redis 服务器的并发量,但是在使用 Redis 读写分离时,也需要注意延迟不一致、数据过期问题。</p>
7 u% g' E. v, _4 i' O |<h4 id="321-延迟与不一致问题">3.2.1 延迟与不一致问题</h4>
) F" G6 y: ~( W2 U4 q1 ~/ A<p>对于命令传播阶段:因为命令传播阶段是异步操作,所以延迟与数据的不一致无法避免。有以下解决方式:</p>2 A' U0 S3 w0 m* |1 V
<ul>
( T, K; K" S6 W<li>若应用对数据不一致的接受程度较低,可以优化中从节点之间的网络环境、使用集群同时扩展写负载和读负载、监控主从节点延迟(offset)判断,若从节点延迟过大,则通知应用不再通过该从节点读取数据</li>
( B2 h% t; w; F4 t7 J9 G; T1 K</ul># {, n( z) f% P. e
<p>对于非命令传播的其他阶段,可以对 <code>slave-serve-stale-date</code> 设置为 no 。则从节点只能响应 info、slaveof 等少数命令,可以保证对数据的一致性。</p>9 O) E; [3 s8 g5 o* x' k
<h4 id="322-数据过期问题">3.2.2 数据过期问题</h4>
^+ |0 B0 C& w2 y<p>数据过期问题已经在<a href="https://www.cnblogs.com/EthanWong/p/15866054.html"><em>Redis 的键管理</em></a> 中提到过,在单机 Redis 中存在惰性删除和定期删除两种删除策略。而在主从复制场景下,从库不会主动删除数据,主要通过主库控制从库中过期数据的删除。而主库的删除策略都不能保证主库及时对过期数据执行删除操作,所以当客户端通过 Redis 从库读取数据时很容易读取到已经过期的数据。</p>
1 R2 b5 X9 X9 w" Z1 [ W6 k& A: S<h4 id="323-故障切换问题">3.2.3 故障切换问题</h4>* Q. K) |" G: \# L& [# H" M" h4 m
<p>在没有使用哨兵的读写分离场景下,建议写监控程序进行切换读写分别连接的 Redis 节点。针对于手动进行切换的方式更复杂但是不容易出错。</p>6 F5 Q3 a( H1 y7 c# K
<h4 id="总结">总结</h4>( E6 e( U& |/ g! B$ L2 Y' ]
<blockquote>) H! q0 G2 [; ]* |) Z
<p>在使用读写分离前,可以考虑其他方法增加Redis的读负载能力:如尽量优化主节点(减少慢查询、减少持久化等其他情况带来的阻塞等)提高负载能力;使用Redis集群同时提高读负载能力和写负载能力等。如果使用读写分离,可以使用哨兵,使主从节点的故障切换尽可能自动化,并减少对应用程序的侵入。</p>3 v( D: _1 h+ Q- q
</blockquote># j! G* V% K# _( ?6 u
<h2 id="参考资料">参考资料</h2>! n9 U( O1 E8 j9 i: u4 d: @
<p>《Redis 设计与实现》</p>
4 {% z. \2 g1 D6 y9 Z<p>《Redis 开发与运维》</p>
! F* K0 E8 a5 d<p><a href="https://pdai.tech/md/db/nosql-redis/db-redis-x-copy.html">https://pdai.tech/md/db/nosql-redis/db-redis-x-copy.html</a></p>
" K2 [1 X( C1 `$ z0 U# Q5 W<p><a href="https://time.geekbang.org/column/article/272852">https://time.geekbang.org/column/article/272852</a></p>
. C4 u c; I' I; ~: N* q' ?8 ~4 |<p><a href="https://kaiwu.lagou.com/course/courseInfo.htm?courseId=59#/detail/pc?id=1782">https://kaiwu.lagou.com/course/courseInfo.htm?courseId=59#/detail/pc?id=1782</a></p>* V/ f$ k9 _( l4 i+ @" ?* x
' M4 l) L1 y/ _7 i- N' o) Q |
|