十二. Redis 集群操作配置(超详细配图,配截图详细说明)
@
目录
- [十二. Redis 集群操作配置(超详细配图,配截图详细说明)](#十二. Redis 集群操作配置(超详细配图,配截图详细说明))
- [1. 为什么需要集群-高可用性](#1. 为什么需要集群-高可用性)
- [2. 集群概述(及其搭建)](#2. 集群概述(及其搭建))
- [3. Redis 集群的使用](#3. Redis 集群的使用)
- [4. Redis 集群故障恢复](#4. Redis 集群故障恢复)
- [5. Redis 集群的 Jedis 开发(使用Java程序连接 Redis 同时开启集群)](#5. Redis 集群的 Jedis 开发(使用Java程序连接 Redis 同时开启集群))
- [6. Redis 集群的优缺点](#6. Redis 集群的优缺点)
- [7. 补充:](#7. 补充:)
- [8. 最后:](#8. 最后:)
1. 为什么需要集群-高可用性
为什么需要集群-高可用性:
- 生产环境的实际需求和问题:
- 容量不够,redis 如何进行扩容。
- 并发写操作,redis 如何分摊。
- 主从模式,薪火相传模式,主机宕机,会导致 ip 地址发生变化,应用程序中配置需要修改对应的主机地址,端口等信息。
- 传统解决方案 代理主机来解决
data:image/s3,"s3://crabby-images/0d250/0d250bc5843ed7391215a264fc700548352fabc1" alt=""
上图解图:
- 客户端请求先到代理服务器
- 由代理服务器进行请求转发到对应的业务处理器
- 为了高可用,代理服务,A服务,B服务,C服务都需要搭建主从结构(至少是一主一从 这样就需求搭建至少 8 台服务器)。
- 这种方案的缺点是:成本高,维护困难,如果是一主多从,成本就会更高。
redis3.0 提供解决方案 无中心化集群配置:
data:image/s3,"s3://crabby-images/ea61d/ea61da6736d823fb0c9455986db0ec3679724b52" alt=""
- 各个 Redis 服务仍然采用主从结构。
- 各个 Redis 服务是连通的,任何一台服务器,都可以作为请求入口 。
- 各个 Redis 服务器因为是连通的,可以进行请求转发
- 这种方式,就无中心化 集群配置,可以看到,只需要 6 台服务器即可搞定。
- 无中心化集群配置 ,还会根据 key 值,计算 slot ,把数据分散到不同的主机,从而缓解单个主机的存取压力
- Redis 推荐使用无中心化集群配置。
- 在实际生成环境,各个 Redis 服务器,应当部署到不同的机器(防止机器宕机,主从复制失效)。
2. 集群概述(及其搭建)
- Redis 集群实现了对 Redis 的水平扩容,即启动 N 个 Redis 节点,将整个数据库分布存储在这个 N 个节点中,每个节点存储总数居的 1 / N
- Redis 集群通过分区(partition) 来提供一定程度的可用性(availability) ,即使集群中有一部分节点失效或者无法进行通讯,集群也可以继续处理命令请求。
Redis 集群搭建:实操演示:
data:image/s3,"s3://crabby-images/4d1b0/4d1b0c7f1ff94ca7b2a52612f3763c11f4929c5f" alt=""
redis.conf
配置修改
properties
cluster-enabled yes 打开集群模式
cluster-config-file nodes-6379.conf 设定节点配置文件名
cluster-node-timeout 15000 设定节点失联时间,超过该时间(毫秒),集群自动进行主 从切换
vi /rainbowsea/redis6379.conf
, 删除不必要的内容 增加 cluster 配置, 文件最后内容,如图
properties
include /rainbowsea/redis.conf
pidfile "/var/run/redis_6379.pid"
port 6379
dbfilename "dump6379.rdb"
masterauth rainbowsea
cluster-enabled yes
cluster-config-file nodes-6379.conf
cluster-node-timeout 15000
data:image/s3,"s3://crabby-images/7b6bf/7b6bf6960e29c64a126242b7658ddc5b933e5047" alt=""
data:image/s3,"s3://crabby-images/4549b/4549b933887a38070119dbaabcd3b49302b1d50f" alt=""
sh
[root@localhost rainbowsea]# cp redis6379.conf redis6380.conf
[root@localhost rainbowsea]# cp redis6379.conf redis6381.conf
[root@localhost rainbowsea]# cp redis6379.conf redis6389.conf
[root@localhost rainbowsea]# cp redis6379.conf redis6390.conf
[root@localhost rainbowsea]# cp redis6379.conf redis6391.conf
[root@localhost rainbowsea]#
data:image/s3,"s3://crabby-images/3b287/3b28736acd4a72c890c5c9e20852218de2d2c522" alt=""
data:image/s3,"s3://crabby-images/2f18e/2f18e7e38e779ae2b026652ddbf17879310d4387" alt=""
- 使用查找替换修改另外 5 个文件
data:image/s3,"s3://crabby-images/d37c1/d37c1bce195b885d9e6942434754f7f27348c2d8" alt=""
data:image/s3,"s3://crabby-images/a34d3/a34d33fae4dccaf4fcb26f4fbd04044eb432981d" alt=""
sh
换指令 :%s/6379/6380
其它几个文件以此操作即可, 操作的时候,一定要小心, 最后建议再检查一下
data:image/s3,"s3://crabby-images/1bc26/1bc2604975be188d2a8a789b708b784f2d81f161" alt=""
所有的都要加上这个 masterauth rainbowsea
加上 Redis 的密码,没有设置密码的则不用配置这个。
所有的都要加上这个 masterauth rainbowsea
加上 Redis 的密码,没有设置密码的则不用配置这个。
所有的都要加上这个 masterauth rainbowsea
加上 Redis 的密码,没有设置密码的则不用配置这个。
properties
include /rainbowsea/redis.conf
pidfile "/var/run/redis_6379.pid"
port 6379
dbfilename "dump6379.rdb"
masterauth rainbowsea
cluster-enabled yes
cluster-config-file nodes-6379.conf
cluster-node-timeout 15000
- 启动 6 个 Redis 服务
sh
[root@localhost rainbowsea]# redis-server /rainbowsea/redis6379.conf
[root@localhost rainbowsea]# redis-server /rainbowsea/redis6380.conf
[root@localhost rainbowsea]# redis-server /rainbowsea/redis6381.conf
[root@localhost rainbowsea]# redis-server /rainbowsea/redis6389.conf
[root@localhost rainbowsea]# redis-server /rainbowsea/redis6390.conf
[root@localhost rainbowsea]# redis-server /rainbowsea/redis6391.conf
[root@localhost rainbowsea]# ps -aux | grep redis
data:image/s3,"s3://crabby-images/18808/188084f86acb68762d020ddb8b05270c3ac6365a" alt=""
data:image/s3,"s3://crabby-images/6b430/6b430b065f9a365896fd4b670b2bde58344b6a69" alt=""
- 将六个节点合成一个集群
data:image/s3,"s3://crabby-images/6d2a7/6d2a743d43c45b348af3b61d780cd71660e61a6b" alt=""
data:image/s3,"s3://crabby-images/af7c4/af7c402075e24a7799b94662b4268fb68c8697d0" alt=""
进入到该路径下后,将六个节点合成一个集群的指令:
如下这个是 Redis 没有配置密码的,指令
sh
redis-cli --cluster create --cluster-replicas 1 192.168.76.147:6379 192.168.76.147:6380 192.168.76.147:6381 192.168.76.147:6389 192.168.76.147:6390 192.168.76.147:6391
如下这个是 Redis 配置了密码的,指令
sh
redis-cli --cluster create -a rainbowsea --cluster-replicas 1 192.168.76.147:6379 192.168.76.147:6380 192.168.76.147:6381 192.168.76.147:6389 192.168.76.147:6390 192.168.76.147:6391
注意事项和细节:
- 组合之前,确保所有(你要使用上的端口的) Redis服务器都是启动的,同时在 root 目录下(我这里是 root 配置的)
nodes-xxxx.conf
文件都生成正常。 - 此时不可以用 127.0.0.1 ,需要使用真实的 IP地址(就是你连接 Linux 的地址,Linux 当中使用
ifconfig
指令查询到的地址),在真实生产环境 IP都是独立的。 replicas 1
采用最简单的方式配置集群,一台主机,一台从机,正好三组。- 搭建加群如果没有成功,把
sentinel
进程关闭掉,再试一下。 - 分许主从对应关系。
data:image/s3,"s3://crabby-images/9639d/9639d251e5f7e75a77c5df37deb25cb5bec7ff60" alt=""
data:image/s3,"s3://crabby-images/e186e/e186eafc059d2736218e2a501690796b82661f07" alt=""
- 分析主从对应关系:如下
data:image/s3,"s3://crabby-images/24645/246459bcd8a4a9e8cd0018cd97803edd625f07a4" alt=""
data:image/s3,"s3://crabby-images/7f4d3/7f4d3d314a0c4552e385fbeebb750b14d61267e0" alt=""
- 集群方式登录:
指令: redis-cli -c -p 6379
指令: cluster nodes 命令查看集群信息, 主从的对应关系, 主要看这里我标注的颜色
sh
[root@localhost src]# redis-cli -c -p 6379
127.0.0.1:6379> auth rainbowsea
127.0.0.1:6379> cluster nodes
data:image/s3,"s3://crabby-images/21e26/21e26f47948883a511a6c3181d2d7c529f912bbf" alt=""
data:image/s3,"s3://crabby-images/848b6/848b6c1cee9192a03f37ca413c707e63121dc050" alt=""
注意事项和细节:
sh
[root@localhost src]# redis-cli -c -p 6379
data:image/s3,"s3://crabby-images/49863/49863598f12f25177edaf12c5e80fb678f228fcd" alt=""
- 一个集群至少要有三个主节点。
- 选项
--cluster-replicas 1
表示我们希望为集群中的每个主节点创建一个从节点。 - 分配原则: 尽量保证主服务器和从服务器各自运行在不同的 IP 地址(机器),防止机器故障导致主从机制失效,高可用性得不到保障。
3. Redis 集群的使用
什么是 slots:
Redis 集群启动后, 你会看到如下提示:
- 一个 Redis 集群包含了
16384
个插槽(hash slot) ,编号从0-16383
,Redis 中的每个键都属于这 16384 个插槽的其中一个。注意:这里虽然只有 16384个插槽,但是并不是只能插入 16384个键,多个不同的键可以插入到同一个插槽的,并不是一个插槽一个键的 。 - 集群使用公式
CRC16(key) % 16384
来计算键 key 属于哪个槽,其中 CRC16(key) 语句用于计算键 key 的 CRC16的校验和
data:image/s3,"s3://crabby-images/3a8e2/3a8e2c49e7206df9517cbdc645362650dc6dda9a" alt=""
- 集群中的每个节点负责处理一部分插槽。举个例子:如果一个集群可以有主节点,其中
- 节点 A 负责处理
0号 ~ 5460号
插槽- 节点 B 负责处理
5461号 ~ 10922号
插槽- 节点 C 负责处理
10923号 ~ 16383号
插槽
在集群中录入值:
- 在 Redis 每次录入,查询键值,redis 都会计算出该 key 应该送往的插槽,如果不是该客户端对应服务器的插槽,redis 会告知前往的 Redis 实例地址和端口。
- Redis-cli 客户端提供了
-c
参数实现自动重定向。 - 如
redis-cli -c -p 6379
登入后,再录入,查询键值对可以自动重定向
data:image/s3,"s3://crabby-images/a416b/a416b0b2da871344e694dd041fc9b6ce5cbcb1f1" alt=""
data:image/s3,"s3://crabby-images/91a29/91a298be8b84c57d16cbd281e131413befc0263f" alt=""
data:image/s3,"s3://crabby-images/bb585/bb58508c797e3810b58d19b04c007eb64ebbf132" alt=""
data:image/s3,"s3://crabby-images/a40da/a40daa85b35d93d0ac7f6bd917bcf6ad56c07f32" alt=""
- 不在一个 slot 下的键值,是不能使用 mget,mset 等多键操作。
sh
192.168.76.147:6381> mset k1 "v1" k2 "v2" k3 "v3"
data:image/s3,"s3://crabby-images/d9edb/d9edba55007a3e1b3ebaf8e379e45b4b312e3cc5" alt=""
data:image/s3,"s3://crabby-images/4c816/4c816207c6b467d9d0cf9dca529a3a2ccc7dbc4a" alt=""
- 可以通过
{}
来定义组的概念,从而使 key 中{}
内相同内容的键值对放到一个slot
中去,就解决了上面 mget 分布到不同 slot 而导致失败的原因。
sh
192.168.76.147:6381> mest k1{order} "v1" k2{order} "v2" k3{order} "v3"
data:image/s3,"s3://crabby-images/343e5/343e56f1e43da5034aaade137243c7a1a2278799" alt=""
注意:你如果对键加上了{}组,那么你想要获取到该值的时候,也是要加上对应的{}组的,才能获取到的。
data:image/s3,"s3://crabby-images/f4ee3/f4ee305e48dfa4444e5ecc93770fcbd0d4a88e98" alt=""
data:image/s3,"s3://crabby-images/38462/38462b30d241c86a9e687121e1d846959d493fd8" alt=""
data:image/s3,"s3://crabby-images/6e3f5/6e3f5263747ee9a33434fb8e5e5a228af8c8fa52" alt=""
查询集群中的值:
- 指令:
CLUSTER KEYSLOT <key>
返回 key 对应的 slot 值
sh
192.168.76.147:6381> cluster keyslot k1
data:image/s3,"s3://crabby-images/a5c96/a5c966194d5a655b0af07d7ba404e84601af274e" alt=""
sh
192.168.76.147:6381> cluster keyslot k2{order}
data:image/s3,"s3://crabby-images/a6879/a6879db98dc2881a35f490205515ee31195a565f" alt=""
可以看到归属于
{}
同一组的,Redis都是分配到了同一个 slot 插槽数值当中。
- 指令:
CLUSTER COUNTKEYSINSLOT <slot>
返回 slot 有多少个 key
sh
192.168.76.147:6381> cluster countkeysinslot 12706
(integer) 1
192.168.76.147:6381> cluster countkeysinslot 16025
(integer) 3
data:image/s3,"s3://crabby-images/464da/464da552ab9238a63bfa2a9073eeb36ee34cf738" alt=""
- 指令:
CLUSTER GETKEYSINSLOT <slot><count>
返回 count 个 slot 槽中的键
sh
192.168.76.147:6381> cluster getkeysinslot 16025 1
1) "k1{order}"
192.168.76.147:6381> cluster getkeysinslot 16025 2
1) "k1{order}"
2) "k2{order}"
192.168.76.147:6381> cluster getkeysinslot 16025 3
data:image/s3,"s3://crabby-images/0431b/0431bfed35c07adcc4574e8a5fb62b69909cc3a8" alt=""
data:image/s3,"s3://crabby-images/b0fd3/b0fd3525f9222937ae48b35f9c6be996a7036822" alt=""
4. Redis 集群故障恢复
- 如果主节点下线, 从节点会自动升为主节点(注意 15 秒超时, 再观察比较准确)
sh
[root@localhost ~]# redis-cli -c -p 6380
data:image/s3,"s3://crabby-images/e8ae9/e8ae9a2e1d62457e06b410c4f23d05be0acb6d31" alt=""
data:image/s3,"s3://crabby-images/d4441/d4441d134f5dd2898874197dbff2a630c847bfff" alt=""
data:image/s3,"s3://crabby-images/6902e/6902e388ea6e9973db2814a6930c9ec24bd022a2" alt=""
这里我们将 6380 主机关闭了。
data:image/s3,"s3://crabby-images/9b3d1/9b3d10783adcfd0cb992faf4793e9ad47d6ae4e4" alt=""
data:image/s3,"s3://crabby-images/082ae/082ae21b00b0e576ad57d2eeea76537b56491449" alt=""
- 主节点恢复后,主节点回来变成从机
data:image/s3,"s3://crabby-images/594a0/594a0a25b2b6fc98d7c36e5400c0772ac3b804fd" alt=""
data:image/s3,"s3://crabby-images/13e5b/13e5b4fb74a971bf955a95c5da647c6a6493432d" alt=""
如果所有某一段插槽的主从节点都宕掉了
,Redis 服务是否还能继续,要根据不同的配置而言。
- 如果某一段插槽的主从 都宕机了,而在 redis.conf 配置文件当中
cluster-require-full-coverage
为yes
,那么,整个集群都会被宕掉,无法使用。- 如果某一段插槽的主从 都宕机了,而在 redis.conf 配置文件当中
cluster-require-full-coverage
为no
,那么,仅仅只是该段插槽的数据不能使用了,也无法存储了,其他插槽的数据还可以继续使用。- redis.conf 文件当中的参数
cluster-require-full-coverage
5. Redis 集群的 Jedis 开发(使用Java程序连接 Redis 同时开启集群)
- 即使连接的不是主机,集群会自动切换主机进行存储,主机写,从机读 。
- 无中心化主从集群,无论从哪台主机写的数据,其他主机上都能读到数据。
- 注意:需要将 Redis 相关的端口都打开 否则会报错
配置防火墙将所有相关 Redis 的端口都打开。
sh
[root@localhost src]# firewall-cmd --add-port=6379/tcp --permanent
Warning: ALREADY_ENABLED: 6379:tcp
success
[root@localhost src]# firewall-cmd --add-port=6380/tcp --permanent
success
[root@localhost src]# firewall-cmd --add-port=6381/tcp --permanent
success
[root@localhost src]# firewall-cmd --add-port=6389/tcp --permanent
success
[root@localhost src]# firewall-cmd --add-port=6390/tcp --permanent
success
[root@localhost src]# firewall-cmd --add-port=6391/tcp --permanent
data:image/s3,"s3://crabby-images/d43cf/d43cf59c27ce7af859050364061b686b77bb35f3" alt=""
sh
[root@localhost src]# firewall-cmd --reload
data:image/s3,"s3://crabby-images/8125e/8125e780cafb54c702ea409fe1978f728b7c7fa9" alt=""
sh
[root@localhost src]# firewall-cmd --list-all
data:image/s3,"s3://crabby-images/fa0ac/fa0ac92046328d2a4e099b2bb553cefc79ff64a4" alt=""
在 pom.xml
当中引入 redis.clients
依赖。如下:
xml
<!-- 引入 jedis 依赖-->
<dependency>
<groupId>redis.clients</groupId>
<artifactId>jedis</artifactId>
<version>3.2.0</version>
</dependency>
data:image/s3,"s3://crabby-images/e3f8a/e3f8a548899fa5a6793e3ecec4998137200bdfe7" alt=""
首先测试,是否可以连接到 Redis 服务器。
java
package com.rainbowsea.jedis;
import org.junit.Test;
import redis.clients.jedis.Jedis;
public class JedisCluster_ {
@Test
public void con() {
// 使用 ip地址 + redis的端口的构造器方法
Jedis jedis = new Jedis("192.168.76.147", 6379);
// 如果Redis 配置了密码,则需要进行身份校验
jedis.auth("rainbowsea");
String ping = jedis.ping();
System.out.println("连接成功 ping 返回的结果 = " + ping);
jedis.close(); // 关闭当前连接,注意并没有关闭 Redis
}
}
data:image/s3,"s3://crabby-images/f70d4/f70d4ec6f7df7eb2d9e93da3a80e4d2c05013395" alt=""
data:image/s3,"s3://crabby-images/57737/57737b1fed9adc8fad90354055e83952da96a093" alt=""
java
import org.junit.Test;
import redis.clients.jedis.HostAndPort;
import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisCluster;
import redis.clients.jedis.JedisPoolConfig;
import java.util.HashSet;
import java.util.Set;
public class JedisCluster_ {
public static void main(String[] args) {
Set<HostAndPort> set = new HashSet<>();
set.add(new HostAndPort("192.168.76.147", 6379));
JedisPoolConfig jedisPoolConfig = new JedisPoolConfig();
// 对连接池进行配置
jedisPoolConfig.setMaxTotal(200);
jedisPoolConfig.setMaxIdle(32);
jedisPoolConfig.setMaxWaitMillis(60 * 1000); // 单位是毫秒
jedisPoolConfig.setBlockWhenExhausted(true);
jedisPoolConfig.setTestOnBorrow(true);
JedisCluster jedisCluster = new JedisCluster(set,5000,5000,5,"rainbowsea",jedisPoolConfig );
jedisCluster.set("address", "bj");
String address = jedisCluster.get("address");
System.out.println("address=>" + address);
jedisCluster.close();
}
}
data:image/s3,"s3://crabby-images/b4842/b484238df5d414ce86d820ba9730b04deb198a4c" alt=""
data:image/s3,"s3://crabby-images/2b948/2b9480f5bfde2cdabbc2542c9f0155383e31c86b" alt=""
data:image/s3,"s3://crabby-images/7c5ac/7c5ac7253ed1be2c8b6766b60296a3ecad302d93" alt=""
6. Redis 集群的优缺点
优点:
- 实现扩容。
- 分摊压力。
- 无中心配置相对简单。
缺点:
- 多键操作是不被支持的。
- 多键的 Redis 事务是不被支持的。 lua 脚本不被支持
- 由于集群方案出现较晚,很多公司已经采用了其他的集群方案,而其它方案想要迁移至 redis cluster ,需要整体迁移而不是逐步过渡,复杂度较大。
7. 补充:
将 root 目录下的,rdb、aof 文件都删除掉
data:image/s3,"s3://crabby-images/5c5ec/5c5ec87ea1e54f1d8f5a538427236fb7cb565c98" alt=""
sh
[root@localhost ~]# rm -f dump*.rdb
data:image/s3,"s3://crabby-images/ab20f/ab20fb1968df5451e55b8a5d25d656ca493f7c49" alt=""
data:image/s3,"s3://crabby-images/74e04/74e04688d9cdc285e590ddc026d946ffefecd387" alt=""
8. 最后:
"在这个最后的篇章中,我要表达我对每一位读者的感激之情。你们的关注和回复是我创作的动力源泉,我从你们身上吸取了无尽的灵感与勇气。我会将你们的鼓励留在心底,继续在其他的领域奋斗。感谢你们,我们总会在某个时刻再次相遇。"