在之前的文章中,使用docker完成了redis一主二从三哨兵的设置,但有出现一个新的问题。
哨兵模式中如果主从中master宕机了,是通过哨兵来选举出新的master,在这个选举切换主从的过程,整个redis服务是不可用的。而且哨兵模式中只有一个主节点对外提供服务,因此没法支持更高的并发。而且当个主节点的内存设置也不宜过大。否则会导致持久化文件过大,影响数据恢复或主从同步的效率。
redis集群是由一系列的主从节点群组成的分布式服务器群,它具有复制、高可用和分片特性。Redis集群不需要 sentinel哨兵也能完成节点移除和故障转移的功能。需要将每个节点设置成集群模式,这种集群模式没有中心节点,客户端通过CRC16算法对key进行hash得到一个值,来判断该key存储在哪个主从服务上面,因此就算是某一个主从整个宕机,redis集群也是部分可用的。方便水平扩展,可以根据业务规模可以随时加减配置。据官方文档称可以线性扩展到上万个节点(但是官方推荐不超过1000个节点)。redis集群的性能和高可用性均优于哨兵模式。
docker pull redis
cluster-enabled yes # 开启集群模式
cluster-config-file nodes-6379.conf #集群配置信息存放文件名,每一个节点需要有个配置文件,需要6份,每个节点处于集群的角色都需要告知其他所有节点,彼此知道,这个文件用于存储集群模式下的集群状态等信息,这个问及那是由reids自己维护。如果需要重新创建集群,那么需要将这个文件删掉。
cluster-node-timeout 5000 #节点离线时间限制,到达此值时发起某个主从重新选举master
requirepass 123456 # 设置本机密码
masterauth 123456 # 设置访问别的机器的密码
appendonly yes # 开启aof
在上面redis-start.sh中添加如下命令(这种方式启动时候特别便捷,后续会改成使用docker-compose.yml文件的方式)
docker run -p 6381:6381 -p 16381:16381 --name redis-6381 -v /mydata/redis-cluster/redis-6381.conf:/etc/redis/redis.conf -v /mydata/redis-cluster/data6381:/data -d redis redis-server /etc/redis/redis.conf --appendonly yes;
docker run -p 6380:6380 -p 16380:16380 --name redis-6380 -v /mydata/redis-cluster/redis-6380.conf:/etc/redis/redis.conf -v /mydata/redis-cluster/data6380:/data -d redis redis-server /etc/redis/redis.conf --appendonly yes;
docker run -p 6379:6379 -p 16379:16379 --name redis-6379 -v /mydata/redis-cluster/redis-6379.conf:/etc/redis/redis.conf -v /mydata/redis-cluster/data6379:/data -d redis redis-server /etc/redis/redis.conf --appendonly yes;
docker run -p 6382:6382 -p 16382:16382 --name redis-6382 -v /mydata/redis-cluster/redis-6382.conf:/etc/redis/redis.conf -v /mydata/redis-cluster/data6382:/data -d redis redis-server /etc/redis/redis.conf --appendonly yes;
docker run -p 6383:6383 -p 16383:16383 --name redis-6383 -v /mydata/redis-cluster/redis-6383.conf:/etc/redis/redis.conf -v /mydata/redis-cluster/data6383:/data -d redis redis-server /etc/redis/redis.conf --appendonly yes;
docker run -p 6384:6384 -p 16384:16384 --name redis-6384 -v /mydata/redis-cluster/redis-6384.conf:/etc/redis/redis.conf -v /mydata/redis-cluster/data6384:/data -d redis redis-server /etc/redis/redis.conf --appendonly yes;
在/mydata/redis-cluster目录下执行sh redis-start.sh命令
启动后使用 docker ps -a查看
为了方便调试,可以在编写一个 redis-stop.sh脚本,使用sh redis-stop.sh命令进行批量关闭docker容器
docker stop redis-6379;
docker rm redis-6379;
docker stop redis-6380;
docker rm redis-6380;
docker stop redis-6381;
docker rm redis-6381;
docker stop redis-6382;
docker rm redis-6382;
docker stop redis-6383;
docker rm redis-6383;
docker stop redis-6384;
docker rm redis-6384;
进入任何一个容器
docker exec -it redis-6379 /bin/bash
执行组件集群的命令:
redis-cli -c --cluster create 47.94.93.93:6379 47.94.93.93:6380 47.94.93.93:6381 47.94.93.93:6382 47.94.93.93:6383 47.94.93.93:6384 --cluster-replicas 1 -a 123456
spring:
redis:
password: 123456
cluster:
nodes: 47.94.93.93:6379,47.94.93.93:6380,47.94.93.93:6381,47.94.93.93:6382,47.94.93.93:6383,47.94.93.93:6384
这个是因为在复制redis.conf文件时粗心,未将文件中的配置port端口号和启动命令中的端口号写一致。
原因是因为 redis集群不仅需要开通redis客户端连接的端口,而且需要开通集群总线端口,集群总线端口为redis客户端连接的端口 + 10000,例如redis端口为6379,则集群总线端口为16379,所有服务器的点需要开通redis的客户端连接端口和集群总线端口,当然,如果有安全组,也要放开这些端口。
具体修改:
在启动命令中添加 -p 16379:16379 然后重新启动即可
创建集群的命令中,这个参数 --cluster-replicas 1,参数后面的数字表示的是主从比例,比如这里的 1 表示的是主从比例是 1:1,也就是 1 个主节点对应1个从节点,现有 6 个实例,所以主从分配就是 3 个 master 主节点,3 个 slave 从节点。
主节点最少3个,3个才能保证集群的健壮性。
如果 --cluster-replicas 2 呢?那么主从比例就是 1:2,也就是 1 个主节点对于应 2 个从节点。即:3(master) + 6(slave) = 9个 Redis 实例。
如果不足 9个 Redis 实例,但是参数指定为 2 会怎么样?
提示已经很清楚了,Redis集群至少需要3个主节点。那么从节点就需要有6个,所以最后说:至少需要9个节点。