Skip to content

about

前面介绍Redis,我们都在一台服务器上进行操作的,也就是说读和写以及备份操作都是在一台Redis服务器上进行的,那么随着项目访问量的增加,对Redis服务器的操作也越加频繁,虽然Redis读写速度都很快,但是一定程度上也会造成一定的延时,那么为了解决访问量大的问题,通常会采取的一种方式是主从架构Master/Slave,Master以写为主,Slave以读为主,Master主节点的数据更新后,自动同步到Slave从节点。

快速部署第二台服务器

PS:第一台服务器的部署参考我的redis安装篇的博客

我的第二台服务器仍然和第一台服务器软硬件配置都是一致的,同样是centos7系统。

那么你可以这么做,直接将第一台服务器的Redis直接拷贝一份,然后稍作更改即可使用。

因为需要两台Linux服务器之间进行数据交互,所以要先配置下免密登录。

此时,你的db02服务器处于启动状态,并且关闭了防火墙。

bash
# 关闭防火墙和下载一些可能用到的工具
systemctl stop firewalld.service
systemctl disable firewalld.service
systemctl status firewalld.service
sed -i.ori 's#SELINUX=enforcing#SELINUX=disabled#g' /etc/selinux/config
yum update -y
yum -y install gcc automake autoconf libtool make
yum -y install net-tools vim wget lrzsz


# 生成类型rsa免密的公钥,下面的命令执行完,一路回车即可
ssh-keygen

[root@cs ~]# ssh-keygen
Generating public/private rsa key pair.
Enter file in which to save the key (/root/.ssh/id_rsa): 
Created directory '/root/.ssh'.
Enter passphrase (empty for no passphrase): 
Enter same passphrase again: 
Your identification has been saved in /root/.ssh/id_rsa.
Your public key has been saved in /root/.ssh/id_rsa.pub.
The key fingerprint is:
SHA256:jRH3eoOMBgLjk4JT5ii5YluscvuSHg8+/dPVSOCLOo8 root@cs
The key's randomart image is:
+---[RSA 2048]----+
|  =     . .      |
|.B +    .o .     |
|B = . ....  .    |
|.+.. . ..*.o     |
|o. o   .So=oo    |
|o +   ... o...   |
|.o+o . . .       |
|.o+=+.. .        |
| .++E+o.         |
+----[SHA256]-----+


# 将公钥拷贝到目标机器上
ssh-copy-id 目标ip

[root@cs ~]# ssh-copy-id 192.168.10.150
/usr/bin/ssh-copy-id: INFO: Source of key(s) to be installed: "/root/.ssh/id_rsa.pub"
The authenticity of host '192.168.10.150 (192.168.10.150)' can't be established.
ECDSA key fingerprint is SHA256:IpkQ09hlQlGw6zgKwI8tVRbme+jRTATj6vnOv9a1JaQ.
ECDSA key fingerprint is MD5:7d:8b:62:ca:c2:d0:6a:f5:d0:06:08:22:22:b2:65:d0.
Are you sure you want to continue connecting (yes/no)? yes					# 提示这里输入yes
/usr/bin/ssh-copy-id: INFO: attempting to log in with the new key(s), to filter out any that are already installed
/usr/bin/ssh-copy-id: INFO: 1 key(s) remain to be installed -- if you are prompted now it is to install the new keys
root@192.168.10.150's password:  				# 输入目标机器的登录密码

Number of key(s) added: 1

Now try logging into the machine, with:   "ssh '192.168.10.150'"
and check to make sure that only the key(s) you wanted were added.

接下来,两台机器就可以无密码传输数据了。

这里用到了rsync工具,rsync(remote synchronize)是Liunx/Unix下的一个远程数据同步工具。

在db02上操作:

bash
rsync -avz 192.168.10.150:/opt/redis* /opt/
rsync -avz 192.168.10.150:/usr/local/bin/redis* /usr/local/bin/
rsync -avz 192.168.10.150:/usr/lib/systemd/system/redis.service /usr/lib/systemd/system/

sed -i 's#150#151#g' /opt/redis6379/conf/redis6379.conf
mkdir -p /data/redis6379
groupadd redis -g 1000
# userdel -r redis
useradd redis -u 1000 -g 1000 -M -s /sbin/nologin
chown -R redis:redis /opt/redis*
chown -R redis:redis /data/redis*
systemctl daemon-reload
systemctl start redis
redis-cli PING

关于用户和用户组的补充说明:

bash
# 添加用户报错也正常,因为我这台测试机器,添加过这个用户和组, 
# -u和-g选项表示同时添加具有特定UID和GID的用户
# -M创建一个没有主目录的用户
# -s表示当前创建的当前用户无法用来登录系统
# chown -R redis:redis表示指定目录以及内部的文件所有用户属组归于redis:redis
# groupdel redis
# cat /etc/group |grep redis

# 如果添加用户和用户组报错的话,可以执行下面的命令把用户和组删掉
# 1. 分别删除用户和组
userdel <用户>
groupdel <用户组名>

# 2. 使用userdel命令同时删除用户和用户组
userdel -r <用户>


# 如果添加用户和组报错,那就把gid后面的1000换个其他的值,比如1002,1003,1008等等
[root@cs ~]# groupadd redis -g 1000
groupadd: GID '1000' already exists

可能的报错汇总:

bash
1. 在db01上执⾏了命令
2. 配置⽂件⾥的密码没删掉
3. 配置⽂件⾥的重命名参数没删掉
4. ⽤户id和组id冲突
5. 没有rsync
6. 拷⻉过来的配置⽂件没有修改IP地址

配置主从复制

从库可以配置多台,比如一主一从,一主多从,并且所有从库配置主从都是一条命令搞定。

此时我的服务器规划:

  • db01,主节点,192.168.10.150
  • db02,从节点,192.168.10.151

1. 现在db01创建一些key

bash
# 清空日志信息和查看日志大小确认下是否清空了
cat /dev/null > /opt/redis6379/logs/redis6379.log
du -sh /opt/redis6379/logs/redis6379.log

# for循环添加一些key
for i in {1..10000}; do redis-cli set zhangkai$i $i;echo $i;done
redis-cli bgsave
redis-cli dbsize

[root@cs opt]# for i in {1..10000}; do redis-cli set zhangkai$i $i;echo $i;done
[root@cs ~]# redis-cli bgsave
Background saving started
[root@cs ~]# redis-cli dbsize
(integer) 10000

ok,一万条数据准备好了,主库这边完事了。

2. db02这边开始操作

PS:保证你的Redis服务成功启动了。

配置主从非常简单,就一条命令就完了。

临时生效,就是直接执行SLAVEOF命令就行了:

bash
# SLAVEOF 主库IP 端口
cat /dev/null > /opt/redis6379/logs/redis6379.log
du -sh /opt/redis6379/logs/redis6379.log
redis-cli flushall
redis-cli SLAVEOF 192.168.10.150 6379
redis-cli dbsize

[root@cs ~]# redis-cli flushall
OK
[root@cs ~]# redis-cli dbsize
(integer) 0
[root@cs ~]# redis-cli SLAVEOF 192.168.10.150 6379
OK
[root@cs ~]# redis-cli dbsize
(integer) 10000

永久生效就是写入到redis配置文件中:

bash
echo "SLAVEOF 192.168.10.150 6379" >> /opt/redis6379/conf/redis6379.conf
cat /opt/redis6379/conf/redis6379.conf

可以看到,只需要一条命令,就立马生效了。

其它命令:

bash
# 查看主从状态的各种信息
INFO REPLICATION
# 查看自己的角色
ROLE

[root@cs ~]# redis-cli INFO REPLICATION
# Replication
role:slave
master_host:192.168.10.150
master_port:6379
master_link_status:up
master_last_io_seconds_ago:1
master_sync_in_progress:0
slave_repl_offset:196
slave_priority:100
slave_read_only:1
connected_slaves:0
master_replid:1c2c8c97d3b3c4e4ba6fc1237b3742161f114cbc
master_replid2:0000000000000000000000000000000000000000
master_repl_offset:196
second_repl_offset:-1
repl_backlog_active:1
repl_backlog_size:1048576
repl_backlog_first_byte_offset:1
repl_backlog_histlen:196
[root@cs ~]# redis-cli ROLE
1) "slave"
2) "192.168.10.150"
3) (integer) 6379
4) "connected"
5) (integer) 196
[root@cs ~]#

主从复制背后的故事

参考:https://zhuanlan.zhihu.com/p/351753935

Replication工作原理

Redis Replication是一种简单、易用的主从模式(master-slave)的复制机制,它能够使得slave节点成为与master节点完全相同的副本。

每次与master节点连接中断后slave节点会自动重联,并且无论master节点发生什么,slave节点总是尝试达到与master节点一致的状态。

Redis采取了一系列的辅助措施来保证数据安全。

Redis主从复制技术有两个版本:在2.8版本之前,每次slave节点断线重联后,只能进行全量同步。在2.8版本之后进行了重新设计,引入了部分同步的概念。本文将以Redis 6.0版本为基础对主从复制原理进行介绍。

旧版主从复制采用的是“全量同步+命令传播”机制完成主从数据同步,这里的硬伤是从库重连后,哪怕主从之间只有少量的数据不一致,也要执行一个耗时、耗资源的全量同步操作来达到数据一致。为此,Redis团队引入了若干机制确保在少量数据不一致时,采用代价较低的部分同步来完成主从复制。所以,当前的主从复制机制包含三个部分:全量同步、部分同步、命令传播。

为保证内容完整性,还是先介绍一下全量同步、命令传播两个历史的概念。

  • 全量同步:master节点创建全量数据的RDB快照文件,通过网络连接发送给slave节点,slave节点加载快照文件恢复数据,然后再继续发送复制过程中复制积压缓冲区内新增的命令,使之达到数据一致状态。
  • 命令传播:如果master-slave节点保持连接,master节点将持续向slave节点发送命令流,以保证master节点数据集发生的改变同样作用在slave节点数据集上,这些命令包含:客户端写请求、key过期、数据淘汰以及其他所有引起数据集变更的操作。

同步原理

在主从复制模式下,Redis使用一对Replicaion ID, offset来唯一识别Master节点数据集的版本,要理解这个“版本“的概念需要认识Redis的以下三个概念:

  • Replication ID(复制ID):每个Redis的主节点都用一个随机生成的字符串来表示在某一时刻其内部存储数据的状态,“某一时刻”可以理解为其成为master角色的那一刻,由源码可知在第一个从节点加入时,Redis初始化了复制ID。
  • offset(复制偏移量):主从模式下,主节点会持续不断的向从节点传播引起数据集更改的命令,offset所表示的是主节点向从节点传递命令字节总数。它不是孤立存在的,需要配合复制积压缓冲区才能工作。
  • backlog(复制积压缓冲区):它是一个环形缓冲区,用来存储主节点向从节点传递的命令,它的大小是固定的,可存储的命令有限,超出部分将会被删除。它即可用于部分同步,也可用于命令传播阶段的命令重推。

来个图,更直观:

我们也能从日志来着手印证下,下面是我按照打印日志的时间顺序整理了下其过程(部分关键日志):

bash
从库:
# 从库向主库发送主从同步请求
54340:S 03 Aug 2023 23:20:30.274 * REPLICAOF 192.168.10.150:6379 enabled (user request from 'id=10 addr=127.0.0.1:33956 fd=7 name= age=0 idle=0 flags=N db=0 sub=0 psub=0 multi=-1 qbuf=48 qbuf-free=32720 obl=0 oll=0 omem=0 events=r cmd=slaveof')


主库:
# 主库接收到了从库的请求
21566:M 03 Aug 2023 15:20:30.667 * Replica 192.168.10.151:6379 asks for synchronization
21566:M 03 Aug 2023 15:20:30.667 * Partial resynchronization not accepted: Replication ID mismatch (Replica asked for '6dd2fe7d7518ca1547f387d5158fd2015c9b39f5', my replication IDs are '113f5106075b494c44df2ff79ac68e7044a6a59b' and '0000000000000000000000000000000000000000')

# 主库立即执行bgsave,拍个最新的快照保存到本地并向从库发送rdb数据
21566:M 03 Aug 2023 15:20:30.667 * Starting BGSAVE for SYNC with target: disk
21566:M 03 Aug 2023 15:20:30.667 * Background saving started by pid 42854
42854:C 03 Aug 2023 15:20:30.672 * DB saved on disk
42854:C 03 Aug 2023 15:20:30.673 * RDB: 0 MB of memory used by copy-on-write
21566:M 03 Aug 2023 15:20:30.774 * Background saving terminated with success
21566:M 03 Aug 2023 15:20:30.779 * Synchronization with replica 192.168.10.151:6379 succeeded

从库:
# 从库接受主库发来的rdb数据
54340:S 03 Aug 2023 23:20:30.834 * Connecting to MASTER 192.168.10.150:6379
54340:S 03 Aug 2023 23:20:30.834 * MASTER <-> REPLICA sync started
54340:S 03 Aug 2023 23:20:30.834 * Non blocking connect for SYNC fired the event.
54340:S 03 Aug 2023 23:20:30.849 * Master replied to PING, replication can continue...
54340:S 03 Aug 2023 23:20:30.851 * Trying a partial resynchronization (request 6dd2fe7d7518ca1547f387d5158fd2015c9b39f5:329).
54340:S 03 Aug 2023 23:20:30.899 * Full resync from master: 113f5106075b494c44df2ff79ac68e7044a6a59b:835852
54340:S 03 Aug 2023 23:20:30.899 * Discarding previously cached master state.
# 接收到了来自主库的168951 bytes
54340:S 03 Aug 2023 23:20:31.004 * MASTER <-> REPLICA sync: receiving 168951 bytes from master
# 重要动作:清空从库自己老的数据,然后加载主库同步过来的rdb数据到内存中
54340:S 03 Aug 2023 23:20:31.010 * MASTER <-> REPLICA sync: Flushing old data
54340:S 03 Aug 2023 23:20:31.010 * MASTER <-> REPLICA sync: Loading DB in memory
54340:S 03 Aug 2023 23:20:31.013 * MASTER <-> REPLICA sync: Finished with success
# 上面已经同步成功了rdb数据,接下来就是同步aof了,执行aof重写机制处理aof文件
54340:S 03 Aug 2023 23:20:31.014 * Background append only file rewriting started by pid 54450
54340:S 03 Aug 2023 23:20:31.092 * AOF rewrite child asks to stop sending diffs.
54450:C 03 Aug 2023 23:20:31.093 * Parent agreed to stop sending diffs. Finalizing AOF...
54450:C 03 Aug 2023 23:20:31.093 * Concatenating 0.00 MB of AOF diff received from parent.
54450:C 03 Aug 2023 23:20:31.093 * SYNC append only file rewrite performed
54450:C 03 Aug 2023 23:20:31.093 * AOF rewrite: 8 MB of memory used by copy-on-write
54340:S 03 Aug 2023 23:20:31.142 * Background AOF rewrite terminated with success
54340:S 03 Aug 2023 23:20:31.142 * Residual parent diff successfully flushed to the rewritten AOF (0.00 MB)
54340:S 03 Aug 2023 23:20:31.142 * Background AOF rewrite finished successfully
# aof搞定之后,主从同步也正式完毕了
# 后续只要在主库的修改操作,都会实时同步到从库

Replication相关其他问题

从库只读

默认情况下,从库工作在只读模式下,即无法对从库执行写指令。比如你在从库服务器上执行:

bash
[root@cs ~]# redis-cli get zhangkai1
"1"
[root@cs ~]# redis-cli set k1 v1
(error) READONLY You can't write against a read only replica.

若要更改此模式,可在配置文件修改如下选项:

bash
# 默认是yes-只读,no-可写
slave-read-only yes/no

但一般,我们从库不负责写入。

过期key处理

Redis可以通过设置key的过期时间来限制key的生存时间,Redis处理key过期有惰性删除和定期删除两种机制,这一机制依赖Redis实例的计时能力。如果主库、从库同时启用key过期的处理机制,可能会导致一些问题。为此,Redis采取了三个技术手段来解决key过期的问题:

  • 从库禁用主动key过期机制。主库在执行key过期后,会以DEL指令的方式向所有从库传播指令,从而保证从库移除过期的key。
  • 依赖主库的key过期机制是无法做到实时性的,所以针对读操作,从库将会按照自己的时钟向客户端返回key不存在。
  • 为防止Lua脚本执行期间key过期,Lua脚本将会传播给从库执行。

心跳机制

在命令传播阶段,每隔一秒,slave节点向master节点发送一次心跳信息,命令格式为REPLCONF ACK <offset>

命令中的offset是就是slave最新的复制偏移量,master接收后便会与自己的offset对比。如果从节点数据缺失,主节点会推送缺失的数据。

min-replica机制

Redis主从复制不仅仅是解决主库、从库之间数据同步的问题,它还需要保证数据的安全性。这里的安全性主要是指主从之间数据同步达到一致的效率,以及主从结构下读写分离场景中分布式系统的可靠性。

Redis采用异步复制机制,它无法真正保证每个从库都能准确的收到传播的指令,所以主从之间必然会存在命令丢失的时间窗口。

为此,Redis引入了min-replicas选项,该机制在redis.conf中有两个配置项:

  • min-replicas-to-write :至少有N个从库才能写入数据。保证从库最低数量。
  • min-replicas-max-lag :如果每个从库的延迟值大于N,则拒绝写入数据。保证主从同步延迟。

通过info replication可以查看从库数量(connected_slaves)、每个从库的延迟值(lag)。

bash
[root@cs ~]# redis-cli INFO REPLICATION
# Replication
role:master
connected_slaves:2
slave0:ip=192.168.10.151,port=6379,state=online,offset=8453211,lag=0
slave1:ip=192.168.10.152,port=6379,state=online,offset=8453211,lag=1
master_replid:1c2c8c97d3b3c4e4ba6fc1237b3742161f114cbc
master_replid2:0000000000000000000000000000000000000000
master_repl_offset:8453211
second_repl_offset:-1
repl_backlog_active:1
repl_backlog_size:1048576
repl_backlog_first_byte_offset:7404636
repl_backlog_histlen:1048576

这一机制是通过从库与主库之间心跳来实现的,如上文所讲,从库每隔一秒向主库发送一次心跳数据,基于心跳,主节点可以:

  • 更新从库同步确认时间:基于主节点时间及同步确认时间计算延迟值。
  • 更新主从最后通信时间:用于从库通信超时检测,如果通信超时,主库会移除从库。

主库关闭持久化时的复制安全性

当master关闭了持久化时,如果发生故障后自动重启时,由本地没有保存持久化的数据,重启的Redis内存数据为空,而slave会自动同步master的数据,就会导致slave的数据也会被清空。

所以,我们应该尽可能为master节点开启持久化,这样可以防止Redis故障重启时数据丢失,进而导致slave数据被清除。如果确实无法开启持久化机制,那应该配置master节点无法自动重启,确保从库可以成为新的master节点,防止数据被清除。

断开主从复制

断开主从复制的状态也非常简单,只要在从库执行下面的命令就好了:

bash
redis-cli SLAVEOF no one

[root@cs opt]# redis-cli SLAVEOF no one
OK

主从复制注意事项

  1. 从节点只读不可写
  2. 从节点不会⾃动故障转移,他会⼀直尝试同步主节点,并且依然不可写
  3. 主从复制故障转移需要介⼊的地⽅ - 修改代码指向新主的IP - 从节点需要执⾏slaveof no one
  4. 从库建⽴同步时会清空⾃⼰的数据,如果同步对象写错了,就清空了
    
  5. 断开主从复制状态后,从库会自动变成主库
    
  6. ⼀定要做好数据备份,⽆论是主节点还是从节点,操作前最好做下备份