主从复制,是指将一台Redis服务器的数据,复制到其他的Redis服务器。
前者称为主节点(Master),后者称为从节点(Slave);数据的复制是单向的,只能由主节点到从节点。
一个master可以有多个slave;
一个slave只能有一个master;
数据流是单向的,master到slave;
主从复制底层依赖与RDB方式进行全量复制。
Redis主从之间的复制分为两点:
Slave 通过 psync命令同步数据与Master建立socket长连接;
Master 收到psync命令,执行bgsave语句生成RDB快照;
Master发送RDB数据;
Slave清空数据并加载Master发来的RDB数据;
Master将生成RDB文件过程中,修改的数据,从repl buffer发送给Slave;
Slave执行接收到的修改命令;
Master通过socket长连接持续把写命令发送给从接单,保证数据一致性。
如果由于网络原因造成原因造成主从断开,期间有数据写入master,再次形成主从时,会形成增量复制。
Slave连接断开;
Master最近数据的修改命令的缓存;
Slave重新建立Socket长连接到master;
Slave psync命令同步数据: offset偏移量控制;
Master判断 Slave的offset。如果在repl backlog buffer中,Master会将缓存中从salve的offset之后数据一次性同步给salve节点,否则会全量同步;
Master通过socket长连接持续把写命令发送给从节点,保证主从一致性。
主从刚刚连接的时候,进行全量同步;全同步结束后,进行增量同步。
当然,如果有需要,slave 在任何时候都可以发起全量同步。
redis 策略是,无论如何,首先会尝试进行增量同步,如不成功,要求从机进行全量同步
节点服务器:
master 192.168.67.104
slave1 192.168.67.103
slave2 192.168.67.102
#所有节点服务器配置
systemctl disable --now firewalld
#开机自动关闭防火墙
sed -i 's/enforcing/disabled/' /etc/selinux/config
#永久关闭selinux策略
所有节点编译安装redis数据库
#修改内核参数
vi /etc/sysctl.conf
vm.overcommit_memory = 1
net.core.somaxconn = 2048
#vm.overcommit_memory:该配置项用于设置内存超额分配策略。
#当值为 0 时,表示内核会根据实际情况动态分配内存,但这可能导致内存超额分配,当系统内存不足时,可能会引发OOM(Out of Memory)错误。
#将其设置为 1 可以禁用内存超额分配,确保只有在内存充足的情况下才能进行内存分配。这对于确保系统稳定性和可靠性是有帮助的。
#net.core.somaxconn:该配置项用于设置系统中允许的最大挂起连接数。
#当系统接收到大量并发连接请求时,如果系统的连接队列已满,则新的连接请求会被拒绝。
#通过增加 net.core.somaxconn 的值,可以增加系统的连接队列大小,从而提高系统处理并发连接请求的能力。
sysctl -p
#刷新参数
wget https://download.redis.io/releases/redis-7.0.13.tar.gz
#可以在官网上下载安装包
redis-7.0.9.tar.gz #所需安装包
yum install -y gcc gcc-c++ make
tar zxvf /opt/redis-7.0.9.tar.gz -C /opt/
cd /opt/redis-7.0.9
make
make PREFIX=/usr/local/redis install #指定安装目录
#由于Redis源码包中直接提供了 Makefile 文件,所以在解压完软件包后,不用先执行 ./configure 进行配置,可直接执行 make 与 make install 命令进行安装
mkdir /usr/local/redis/{conf,log,data}
cp /opt/redis-7.0.9/redis.conf /usr/local/redis/conf/
#建立用户redis来管理服务,更改工作目录的属主属组
useradd -M -s /sbin/nologin redis
chown -R redis.redis /usr/local/redis/
#创建管理用户并更改属主和属组
#添加环境变量
vi /etc/profile
PATH=$PATH:/usr/local/redis/bin
#增加一行
#可以直接使用redis命令
source /etc/profile
#刷新文件
vi /usr/local/redis/conf/redis.conf
bind 127.0.0.1 192.168.67.104 #87行,添加 监听的主机地址
protected-mode no
#111行,将本机访问保护模式设置no。如果开启了,那么在没有设定bind ip且没有设密码的情况下,Redis只允许接受本机的响应
port 6379 #138行,Redis默认的监听6379端口
daemonize yes #309行,设置为守护进程,后台启动
pidfile /usr/local/redis/log/redis_6379.pid #341行,指定 PID 文件
logfile "/usr/local/redis/log/redis_6379.log"
#354行,指定日志文件
dir /usr/local/redis/data #504行,指定持久化文件所在目录
requirepass abc123 #1037行,增加一行,设置redis密码
vi /usr/lib/systemd/system/redis-server.service
[Unit]
Description=Redis Server
After=network.target
[Service]
User=redis
Group=redis
Type=forking
TimeoutSec=0
PIDFile=/usr/local/redis/log/redis_6379.pid
ExecStart=/usr/local/redis/bin/redis-server /usr/local/redis/conf/redis.conf
ExecReload=/bin/kill -s HUP $MAINPID
ExecStop=/bin/kill -s QUIT $MAINPID
PrivateTmp=true
[Install]
WantedBy=multi-user.target
systemctl daemon-reload
#重新加载
#启动服务
systemctl start redis-server
systemctl enable redis-server
ss -lntp | grep 6379
#查看相关进程
vim /usr/local/redis/conf/redis.conf
#修改redis主配置文件
bind 0.0.0.0 #87行,修改监听地址为0.0.0.0
protected-mode no #111行,将本机访问保护模式设置no
port 6379 #138行,Redis默认的监听6379端口
daemonize yes #309行,设置为守护进程,后台启动
pidfile /usr/local/redis/log/redis_6379.pid #341行,指定 PID 文件
logfile "/usr/local/redis/log/redis_6379.log" #354行,指定日志文件
dir /usr/local/redis/data #504行,指定持久化文件所在目录
#requirepass abc123 #1037行,可选,设置redis密码
appendonly yes #1380行,开启AOF
systemctl restart redis-server.service
#重启数据库
slave
vim /usr/local/redis/conf/redis.conf
#编辑redis主配置文件
bind 0.0.0.0
#87行,修改监听地址为0.0.0.0
protected-mode no
#111行,将本机访问保护模式设置no
port 6379
#138行,Redis默认的监听6379端口
daemonize yes
#309行,设置为守护进程,后台启动
pidfile /usr/local/redis/log/redis_6379.pid #341行,指定 PID 文件
logfile "/usr/local/redis/log/redis_6379.log" #354行,指定日志文件
dir /usr/local/redis/data
#504行,指定持久化文件所在目录
replicaof 192.168.67.104 6379
#528行,指定要同步的Master节点IP和端口
masterauth abc123
#535行,可选,指定Master节点的密码,仅在Master节点设置了requirepass
requirepass abc123
#1038行,可选,设置redis密码
appendonly yes
#1380行,开启AOF
systemctl restart redis-server.service
#重启服务
slave2
#在slave1中
scp 192.168.67.102:/usr/local/redis/conf/redis.conf /usr/local/redis/conf/redis.conf
#远程传输配置文件
systemctl restart redis-server.service
#前往master查看日志
tail -f /usr/local/redis/log/redis_6379.log
#在Master节点上验证从节点:
redis-cli -a abc123 info replication -a
主从切换技术的方法是:当服务器宕机后,需要手动一台从机切换为主机,这需要人工干预,不仅费时费力而且还会造成一段时间内服务不可用。
为了解决主从复制的缺点,就有了哨兵机制。
哨兵的核心功能:在主从复制的基础上,哨兵引入了主节点的自动故障转移。
默认端口号:26379
监控:哨兵会不断地检查主节点和从节点是否运作正常。
自动故障转移:当主节点不能正常工作时,哨兵会开始自动故障转移操作,它会将失效主节点的其中一个从节点升级为新的主节点,并让其它从节点改为复制新的主节点。
通知(提醒):哨兵可以将故障转移的结果发送给客户端。
由哨兵节点定期监控发现主节点是否出现了故障
当主节点出现故障,此时哨兵节点会通过Raft算法(选举算法)实现选举机制共同选举出一个哨兵节点为leader,来负责处理主节点的故障转移和通知。
所以整个运行哨兵的集群的数量不得少于3个节点。
由leader哨兵节点执行故障转移
将某一个从节点升级为新的主节点,让其它从节点指向新的主节点;
若原主节点恢复也变成从节点,并指向新的主节点;
通知客户端主节点已经更换。
注:客观下线是主节点才有的概念;如果从节点和哨兵节点发生故障,被哨兵主观下线后,不会再有后续的客观下线和故障转移操作。
#节点服务器
master 192.168.67.104
slave1 192.168.67.103
slave2 192.168.67.102
cp /opt/redis-7.0.9/sentinel.conf /usr/local/redis/conf/
#拷贝模板文件
chown redis.redis /usr/local/redis/conf/sentinel.conf
#更改文件属主和属组
vi /usr/local/redis/conf/sentinel.conf
#编辑配置文件
protected-mode no
#6行,关闭保护模式
port 26379 #10行,Redis哨兵默认的监听端口
daemonize yes #15行,指定sentinel为后台启动
pidfile /usr/local/redis/log/redis-sentinel.pid
#20行,指定 PID 文件
logfile "/usr/local/redis/log/sentinel.log" #25行,指定日志存放路径
dir /usr/local/redis/data #54行,指定数据库存放路径
sentinel monitor mymaster 192.168.67.104 6379 2
#73行
#指定该哨兵节点监控192.168.67.104:6379这个主节点
#该主节点的名称是mymaster,
#最后的2的含义与主节点的故障判定有关:至少需要2个哨兵节点同意,才能判定主节点故障并进行故障转移
sentinel auth-pass mymaster abc123 #76行指定Master节点的密码,仅在Master节点设置了requirepass
sentinel down-after-milliseconds mymaster 3000
#114行,判定服务器down掉的时间周期,默认3000毫秒(3秒)
sentinel failover-timeout mymaster 180000 #214行,同一个sentinel对同一个master两次failover之间的间隔时间(180秒)
#VIP地址漂移
#修改哨兵配置文件279行,添加脚本路径
sentinel client-reconfig-script mymaster /usr/local/redis/conf/fail.sh
#!/bin/bash
MASTER_IP=$6
#表示传递的第六个参数,即新Master的地址
INTERFACE="ens33"
LOCAL_IP=$(ifconfig $INTERFACE | awk 'NR==2 {print $2}')
#本机IP 网卡ens33对应
VIP="192.168.2.200"
#设定VIP
NETMASK="24"
KEY="1"
#判断当前节点是否为主节点
if [ "$MASTER_IP" = "$LOCAL_IP" ];then
/sbin/ifconfig ${INTERFACE}:${KEY} ${VIP}/${NETMASK}
#如果是,将设定的VIP绑定到指定的网络接口上
exit 0
else
/sbin/ifconfig ${INTERFACE}:${KEY} down
#如果不是,将指定的网络接口和绑定的VIP停止
fi
exit 1
chmod +x fail.sh
#给脚本加执行权限
#Master节点的脚本和配置文件传输给Slave1
scp /usr/local/redis/conf/sentinel.conf 192.168.67.103:/usr/local/redis/conf/sentinel.conf
scp /usr/local/redis/conf/fail.sh 192.168.67.103:/usr/local/redis/conf/
#Master节点的脚本和配置文件传输给Slave2
scp /usr/local/redis/conf/sentinel.conf 192.168.67.102:/usr/local/redis/conf/sentinel.conf
scp /usr/local/redis/conf/fail.sh 192.168.67.102:/usr/local/redis/conf/
chown redis.redis /usr/local/redis/conf/sentinel.conf
#复制过后需要更改属主和属组
注:一定要先启动master,再启动slave
cd /usr/local/redis/conf/
redis-sentinel sentinel.conf &
redis-cli -p 26379 info Sentinel
#执行脚本
#添加VIP
bash fail.sh 1 1 1 1 1 192.168.67.104 1
ifconfig
观察各个节点的master、slave状态,还有VIP地址能否自动漂移。
#关闭主节点的Redis服务
systemctl stop redis-server
#通过slave1观察是否自动故障转移
redis-cli -h 192.168.67.103 -p 26379
info sentinel
#前往slave1查看VIP是否漂移
ifconfig
#重新启动master节点redis服务
systemctl start redis-server
#连接到redis
redis-cli -a abc123 -h 192.168.67.104 info replication
#观察两个配置文件
#可以看到配置文件自动被修改,原来的master状态变为slave
集群,即Redis Cluster
,是Redis 3.0开始引入的分布式存储方案。
集群由多组节点(Node)组成,Redis的数据分布在这些节点中。
集群中的节点分为主节点和从节点:
可以归纳为两点
数据分区(或称数据分片)是集群最核心的功能。
集群将数据分散到多个节点,一方面突破了Redis单机内存大小的限制,存储容量大大增加;
另一方面每个主节点都可以对外提供读服务和写服务,大大提高了集群的响应能力。
注:Redis单机内存大小受限问题
例如:
如果单机内存太大,bgsave和bgrewriteaof的fork操作可能导致主进程阻塞,主从环境下主机切换时可能导致从节点长时间无法提供服务,全量复制阶段主节点的复制缓冲区可能溢出。
#以3个节点组成的集群为例:
节点A包含0到5460号哈希槽
节点B包含5461到10922号哈希槽
节点C包含10923到16383号哈希槽
#Redis集群的主从复制模型
集群中具有A、B、C三个节点,如果节点B失败了,整个集群就会因缺少5461-10922这个范围的槽而不可以用。
为每个节点添加一个从节点A1、B1、C1整个集群便有三个Master节点和三个slave节点组成,在节点B失败后,集群选举B1位为的主节点继续服务。当B和B1都失败后,集群将不可用。
方便起见,这里在同一台服务器上模拟。
服务器 | IP | 主端口 | 从端口 |
---|---|---|---|
Node1节点 | 192.168.2.106 | 6001 | 6004 |
Node2节点 | 192.168.2.106 | 6002 | 6005 |
Node3节点 | 192.168.2.106 | 6003 | 6006 |
cd /usr/local/redis/
mkdir -p redis-cluster/redis600{1..6}
#复制 Redis 配置文件和可执行文件到不同目录
for i in {1..6}
do
#执行复制 循环6次
cp /opt/redis-7.0.9/redis.conf /usr/local/redis/redis-cluster/redis600$i
cp /opt/redis-7.0.9/src/redis-cli /opt/redis-7.0.9/src/redis-server /usr/local/redis/redis-cluster/redis600$i
done
#创建相应pid和日志文件
touch redis_6001.pid
touch redis_6001.log
#重复操作
其他5个文件夹的配置文件配置类似,6个端口都要不一样。
cd /usr/local/redis/redis-cluster/redis6001
vim redis.conf
#bind 127.0.0.1 #87行,注释掉bind项,默认监听所有网卡
protected-mode no #111行,关闭保护模式
port 6001 #138行,修改redis监听端口
daemonize yes #309行,设置为守护进程,后台启动
pidfile /usr/local/redis/log/redis_6001.pid #341行,指定 PID 文件
logfile "/usr/local/redis/log/redis_6001.log" #354行,指定日志文件
dir ./ #504行,指定持久化文件所在目录
appendonly yes #1379行,开启AOF
cluster-enabled yes #1576行,取消注释,开启群集功能
cluster-config-file nodes-6001.conf #1584行,取消注释,群集名称文件设置
cluster-node-timeout 15000 #1590行,取消注释群集超时时间设置
#将6001的配置文件,分别复制给2-6
cp redis.conf ../redis6002/
cp redis.conf ../redis6003/
cp redis.conf ../redis6004/
cp redis.conf ../redis6005/
cp redis.conf ../redis6006/
#使用sed,直接替换端口号
sed -i 's/6001/6002/' ../redis6002/redis.conf
#以6002为例,其余操作相同
sed -n '/6002/p' redis.conf
#查看是否更改成功
#进入各个节点文件
redis-server redis.conf
#执行命令,启动节点
ps -ef | grep redis
#查看进程
redis-cli --cluster create 127.0.0.1:6001 127.0.0.1:6002 127.0.0.1:6003 127.0.0.1:6004 127.0.0.1:6005 127.0.0.1:6006 --cluster-replicas 1
#六个实例分为三组,每组一主一从,前面的做主节点,后面的做从节点。下面交互的时候 需要输入 yes 才可以创建。
#--replicas 1 表示每个主节点有1个从节点。
#登录6001
redis-cli -p 6001 -c
#加-c参数,节点之间就可以互相跳转
cluster slots
#查看节点的哈希槽编号范围
#测试
127.0.0.1:6002> set name scj
127.0.0.1:6001> cluster keyslot name
#查看name键的槽编号
redis-cli -p 6005 -c
127.0.0.1:6004> keys * #对应的slave节点也有这条数据,但是别的节点没有
#连接到6001节点并获取集群中的节点信息
redis-cli -p 6001 -c cluster nodes
cd /usr/local/redis/
mkdir -p redis-cluster/redis600{7..8}
#复制 Redis 配置文件和可执行文件到不同目录
for i in {7..8}
do
cp /opt/redis-7.0.9/redis.conf /usr/local/redis/redis-cluster/redis600$i
cp /opt/redis-7.0.9/src/redis-cli /opt/redis-7.0.9/src/redis-server /usr/local/redis/redis-cluster/redis600$i
done
#创建pid及日志文件
touch redis_6007.log
touch redis_6007.pid
#重复操作
cp ../redis6001/redis.conf redis.conf
#覆盖原配置文件
sed -i 's/6001/6007/' redis.conf
#修改其中端口号
sed -n '/6007/p' redis.conf
#重复操作
#修改完成后开启集群功能
./redis-server ./redis.conf
ps -ef | grep redis
#创建一个新的主节点127.0.0.1:6007。命令里需要指定一个已有节点以便于获取集群信息,本例是指定的127.0.0.1:6001
redis-cli -p 6001 --cluster add-node 127.0.0.1:6007 127.0.0.1:6001
或
redis-cli -p 6001
cluster meet 127.0.0.1 6007
cluster meet 127.0.0.1 6008
cluster nodes
#将127.0.0.1:6008创建为127.0.0.1:6007的从节点。命令里需要指定一个已有节点以便于获取集群信息和主节点的node ID
redis-cli -p 6007 --cluster add-node 127.0.0.1:6008 127.0.0.1:6007 --cluster-slave --cluster-master-id 27b680c2a455f25b9a4192fa4bc48e496ef6c5d2
或
redis-cli -p 6008
cluster replicate 27b680c2a455f25b9a4192fa4bc48e496ef6c5d2
#新加入的主节点是没有槽数的,只有初始化集群的时候,才会根据主的数据分配好,如新增的主节点,需要手动分配
redis-cli -p 6007 --cluster reshard 127.0.0.1:6001
#从6001分配哈希槽到6007
#查看集群状态
cluster nodes