惊呆了,竟然可以用这种方式秒建 Redis 集群?

Redis Cluster 是 Redis 3.0 版本推出的 Redis 集群方案,它将数据分布在不同的服务区上,以此来降低系统对单主节点的依赖,并且可以大大的提高 Redis 服务的读写性能。

Redis 将所有的数据分为 16384 个 slots(槽),每个节点负责其中的一部分槽位,当有 Redis 客户端连接集群时,会得到一份集群的槽位配置信息,这样它就可以直接把请求命令发送给对应的节点进行处理。

Redis Cluster 是无代理模式去中心化的运行模式,客户端发送的绝大数命令会直接交给相关节点执行,这样大部分情况请求命令无需转发,或仅转发一次的情况下就能完成请求与响应,所以集群单个节点的性能与单机 Redis 服务器的性能是非常接近的,因此在理论情况下,当水平扩展一倍的主节点就相当于请求处理的性能也提高了一倍,所以 Redis Cluster 的性能是非常高的。

Redis Cluster 架构图如下所示:

惊呆了,竟然可以用这种方式秒建 Redis 集群?

秒建 Redis 集群

Redis Cluster 的搭建方式有两种,一种是使用 Redis 源码中提供的 create-cluster 工具快速的搭建 Redis 集群环境,另一种是配置文件的方式手动创建 Redis 集群环境。

1.快速搭建 Redis Cluster

create-cluster 工具在 utils/create-cluster 目录下,如下图所示:

惊呆了,竟然可以用这种方式秒建 Redis 集群?

使用命令 ./create-cluster start 就可以急速创建一个 Redis 集群,执行如下:

1$ ./create-cluster start# 创建集群2Starting300013Starting300024Starting300035Starting300046Starting300057Starting30006

接下来我们需要把以上创建的 6 个节点节点通过 create 命令组成一个集群,执行如下:

1[@iZ2ze0nc5n41zomzyqtksmZ:create-cluster]$ ./create-cluster create# 组建集群2>>> Performing hash slots allocation on6 nodes... 3Master[0] -> Slots 0 - 54604Master[1] -> Slots 5461 - 10922 5Master[2] -> Slots 10923 - 16383 6Adding replica 127.0.0.1:30005to127.0.0.1:30001 7Adding replica 127.0.0.1:30006to127.0.0.1:30002 8Adding replica 127.0.0.1:30004to127.0.0.1:30003 9>>> Trying tooptimize slaves allocation for anti-affinity10[WARNING] Some slaves arein the same host as their master11M: 445f2a86fe36d397613839d8cc1ae6702c976593 127.0.0.1:3000112 slots:[0-5460] (5461 slots) master13M: 63bb14023c0bf58926738cbf857ea304bff8eb50 127.0.0.1:3000214 slots:[5461-10922] (5462 slots) master15M: 864d4dfe32e3e0b81a64cec8b393bbd26a65cbcc 127.0.0.1:3000316 slots:[10923-16383] (5461 slots) master17S: 64828ab44566fc5ad656e831fd33de87be1387a0 127.0.0.1:3000418 replicates 445f2a86fe36d397613839d8cc1ae6702c97659319S: 0b17b00542706343583aa73149ec5ff63419f140 127.0.0.1:3000520 replicates 63bb14023c0bf58926738cbf857ea304bff8eb5021S: e35f06ca9b700073472d72001a39ea4dfcb541cd 127.0.0.1:3000622 replicates 864d4dfe32e3e0b81a64cec8b393bbd26a65cbcc23Can I set the above configuration? (type'yes'toaccept): yes24>>> Nodes configuration updated25>>> Assign a different config epoch toeach node26>>> Sending CLUSTER MEET messages tojoin the cluster27Waiting for the cluster tojoin28.29>>> Performing Cluster Check (using node 127.0.0.1:30001)30M: 445f2a86fe36d397613839d8cc1ae6702c976593 127.0.0.1:3000131 slots:[0-5460] (5461 slots) master321 additional replica(s)33M: 864d4dfe32e3e0b81a64cec8b393bbd26a65cbcc 127.0.0.1:3000334 slots:[10923-16383] (5461 slots) master351 additional replica(s)36S: e35f06ca9b700073472d72001a39ea4dfcb541cd 127.0.0.1:3000637 slots: (0 slots) slave38 replicates 864d4dfe32e3e0b81a64cec8b393bbd26a65cbcc39S: 0b17b00542706343583aa73149ec5ff63419f140 127.0.0.1:3000540 slots: (0 slots) slave41 replicates 63bb14023c0bf58926738cbf857ea304bff8eb5042M: 63bb14023c0bf58926738cbf857ea304bff8eb50 127.0.0.1:3000243 slots:[5461-10922] (5462 slots) master441 additional replica(s)45S: 64828ab44566fc5ad656e831fd33de87be1387a0 127.0.0.1:3000446 slots: (0 slots) slave47 replicates 445f2a86fe36d397613839d8cc1ae6702c97659348[OK] All nodes agree about slots configuration.49>>> Checkforopen slots...50>>> Check slots coverage...51[OK] All 16384 slots covered.

在执行的过程中会询问你是否通过把 30001、30002、30003 作为为主节点,把 30004、30005、30006 作为它们的从节点,输入 yes 后会执行完成。

我们可以先使用 redis-cli 连接到集群,命令如下:

1$ redis-cli -c -p 30001

在使用 nodes 命令来查看集群的节点信息,命令如下:

1127.0.0.1:30001> clusternodes2864d4dfe32e3e0b81a64cec8b393bbd26a65cbcc 127.0.0.1:30003@40003 master - 015851258350783 connected 10923-163833e35f06ca9b700073472d72001a39ea4dfcb541cd 127.0.0.1:30006@40006 slave 864d4dfe32e3e0b81a64cec8b393bbd26a65cbcc 015851258350786 connected40b17b00542706343583aa73149ec5ff63419f140 127.0.0.1:30005@40005 slave 63bb14023c0bf58926738cbf857ea304bff8eb50 015851258350785 connected563bb14023c0bf58926738cbf857ea304bff8eb50 127.0.0.1:30002@40002 master - 015851258341752 connected 5461-109226445f2a86fe36d397613839d8cc1ae6702c976593 127.0.0.1:30001@40001 myself,master - 015851258350001 connected 0-5460764828ab44566fc5ad656e831fd33de87be1387a0 127.0.0.1:30004@40004 slave 445f2a86fe36d397613839d8cc1ae6702c976593 015851258350004 connected

可以看出 30001、30002、30003 都为主节点,30001 对应的槽位是 0-5460,30002 对应的槽位是 5461-10922,30003 对应的槽位是 10923-16383,总共有槽位 16384 个 (0 ~ 16383)。

30002 对应的槽位是 5461-10922,30003 对应的槽位是 10923-16383,总共有槽位 16384 个 (0 ~ 16383)。

create-cluster 搭建的方式虽然速度很快,但是该方式搭建的集群主从节点数量固定以及槽位分配模式固定,并且安装在同一台服务器上,所以只能用于测试环境。

我们测试完成之后,可以使用以下命令,关闭并清理集群:

1$ ./create-cluster stop# 关闭集群2Stopping 300013Stopping 300024Stopping 300035Stopping 300046Stopping 300057Stopping 300068$ ./create-cluster clean # 清理集群

2.手动搭建 Redis Cluster

由于 create-cluster 本身的限制,在实际生产环境中我们需要使用手动添加配置的方式搭建 Redis 集群,为此我们先要把 Redis 安装包复制到 node1 到 node6 文件中,因为我们要安装 6 个节点,3 主 3 从,如下图所示:

惊呆了,竟然可以用这种方式秒建 Redis 集群?

惊呆了,竟然可以用这种方式秒建 Redis 集群?

接下来我们进行配置并启动 Redis 集群。

① 设置配置文件

我们需要修改每个节点内的 redis.conf 文件,设置 cluster-enabled yes 表示开启集群模式,并且修改各自的端口,我们继续使用 30001 到 30006,通过 port 3000X 设置。

② 启动各个节点

redis.conf 配置好之后,我们就可以启动所有的节点了,命令如下:

1cd /usr/local/soft/mycluster/node12./src/redis-server redis.conf

③ 创建集群并分配槽位

之前我们已经启动了 6 个节点,但这些节点都在各自的集群之内并未互联互通,因此接下来我们需要把这些节点串连成一个集群,并为它们指定对应的槽位,执行命令如下:

1redis-cli--clustercreate 127.0.0.1:30001 127.0.0.1:30002 127.0.0.1:30003 127.0.0.1:30004 127.0.0.1:30005 127.0.0.1:30006--cluster-replicas 1

其中 create 后面跟多个节点,表示把这些节点作为整个集群的节点,而 cluster-replicas 表示给集群中的主节点指定从节点的数量,1 表示为每个主节点设置一个从节点。

在执行了 create 命令之后,系统会为我们指定节点的角色和槽位分配计划,如下所示:

1>>> Performing hash slots allocation on6 nodes... 2Master[0] -> Slots 0 - 54603Master[1] -> Slots 5461 - 10922 4Master[2] -> Slots 10923 - 16383 5Adding replica 127.0.0.1:30005to127.0.0.1:30001 6Adding replica 127.0.0.1:30006to127.0.0.1:30002 7Adding replica 127.0.0.1:30004to127.0.0.1:30003 8>>> Trying to optimize slaves allocation for anti-affinity 9[WARNING] Some slaves are in the same host as their master10M: bdd1c913f87eacbdfeabc71befd0d06c913c891c 127.0.0.1:3000111 slots:[0-5460] (5461 slots) master12M: bdd1c913f87eacbdfeabc71befd0d06c913c891c 127.0.0.1:3000213 slots:[5461-10922] (5462 slots) master14M: bdd1c913f87eacbdfeabc71befd0d06c913c891c 127.0.0.1:3000315 slots:[10923-16383] (5461 slots) master16S: bdd1c913f87eacbdfeabc71befd0d06c913c891c 127.0.0.1:3000417 replicates bdd1c913f87eacbdfeabc71befd0d06c913c891c18S: bdd1c913f87eacbdfeabc71befd0d06c913c891c 127.0.0.1:3000519 replicates bdd1c913f87eacbdfeabc71befd0d06c913c891c20S: bdd1c913f87eacbdfeabc71befd0d06c913c891c 127.0.0.1:3000621 replicates bdd1c913f87eacbdfeabc71befd0d06c913c891c22Can I set the above configuration? (type'yes'to accept):

从以上信息可以看出,Redis 打算把 30001、30002、30003 设置主节点,并为他们分配的槽位,30001 对应的槽位是 0-5460,30002 对应的槽位是 5461-10922,30003 对应的槽位是 10923-16383,并且把 30005 设置为 30001 的从节点、30006 设置为 30002 的从节点、30004 设置为 30003 的从节点,我们只需要输入 yes 即可确认并执行分配,如下所示:

1Can I set the above configuration? (type'yes'toaccept): yes 2>>> Nodes configuration updated 3>>> Assign a different config epoch toeach node 4>>> Sending CLUSTER MEET messages tojoin the cluster 5Waiting for the cluster tojoin6.... 7>>> Performing Cluster Check (using node 127.0.0.1:30001) 8M: 887397e6fefe8ad19ea7569e99f5eb8a803e3785 127.0.0.1:30001 9 slots:[0-5460] (5461 slots) master101 additional replica(s)11S: abec9f98f9c01208ba77346959bc35e8e274b6a3 127.0.0.1:3000512 slots: (0 slots) slave13 replicates 887397e6fefe8ad19ea7569e99f5eb8a803e378514S: 1a324d828430f61be6eaca7eb2a90728dd5049de 127.0.0.1:3000415 slots: (0 slots) slave16 replicates f5958382af41d4e1f5b0217c1413fe19f390b55f17S: dc0702625743c48c75ea935c87813c4060547cef 127.0.0.1:3000618 slots: (0 slots) slave19 replicates 3da35c40c43b457a113b539259f17e7ed616d13d20M: 3da35c40c43b457a113b539259f17e7ed616d13d 127.0.0.1:3000221 slots:[5461-10922] (5462 slots) master221 additional replica(s)23M: f5958382af41d4e1f5b0217c1413fe19f390b55f 127.0.0.1:3000324 slots:[10923-16383] (5461 slots) master251 additional replica(s)26[OK] All nodes agree about slots configuration.27>>> Checkforopen slots...28>>> Check slots coverage...29[OK] All 16384 slots covered.

显示 OK 表示整个集群就已经成功启动了。

接下来,我们使用 redis-cli 连接并测试一下集群的运行状态,代码如下:

1$ redis-cli -c -p 30001 # 连接到集群 2127.0.0.1:30001> cluster info # 查看集群信息3cluster_state:ok # 状态正常 4cluster_slots_assigned:16384 # 槽位数5cluster_slots_ok:16384 # 正常的槽位数 6cluster_slots_pfail:0 7cluster_slots_fail:08cluster_known_nodes:6 # 集群的节点数 9cluster_size:3 # 集群主节点数10cluster_current_epoch:611cluster_my_epoch:112cluster_stats_messages_ping_sent:13013cluster_stats_messages_pong_sent:12714cluster_stats_messages_sent:25715cluster_stats_messages_ping_received:12216cluster_stats_messages_pong_received:13017cluster_stats_messages_meet_received:518cluster_stats_messages_received:257

相关字段的说明已经标识在上述的代码中了,这里就不再赘述。

动态增删节点

某些情况下,我们需要根据实际的业务情况,对已经在运行的集群进行动态的添加或删除节点,那我们就需要进行以下操作。

1.增加主节点

添加方式一:cluster meet

使用 cluster meet ip:port 命令就可以把一个节点加入到集群中,执行命令如下:

1127.0.0.1:30001> clustermeet 127.0.0.1 30007 2OK 3127.0.0.1:30001> clusternodes 4dc0702625743c48c75ea935c87813c4060547cef 127.0.0.1:30006@40006 slave 3da35c40c43b457a113b539259f17e7ed616d13d 015851429160006 connected 5df0190853a53d8e078205d0e2fa56046f20362a7 127.0.0.1:30007@40007 master - 015851429177400 connected6f5958382af41d4e1f5b0217c1413fe19f390b55f 127.0.0.1:30003@40003 master - 015851429167383 connected 10923-1638373da35c40c43b457a113b539259f17e7ed616d13d 127.0.0.1:30002@40002 master - 015851429130002 connected 5461-109228abec9f98f9c01208ba77346959bc35e8e274b6a3 127.0.0.1:30005@40005 slave 887397e6fefe8ad19ea7569e99f5eb8a803e3785 015851429170005 connected9887397e6fefe8ad19ea7569e99f5eb8a803e3785 127.0.0.1:30001@40001 myself,master - 015851429150001 connected 0-5460101a324d828430f61be6eaca7eb2a90728dd5049de 127.0.0.1:30004@40004 slave f5958382af41d4e1f5b0217c1413fe19f390b55f 015851429160004 connected

可以看出端口为 30007 的节点并加入到集群中,并设置成了主节点。

添加方式二:add-node

使用 redis-cli --cluster add-node 添加节点ip:port 集群某节点ip:port 也可以把一个节点添加到集群中,执行命令如下:

1$ redis-cli --cluster add-node 127.0.0.1:30008127.0.0.1:30001 2>>> Adding node 127.0.0.1:30008 to cluster 127.0.0.1:30001 3>>> Performing Cluster Check (using node 127.0.0.1:30001) 4M:887397e6fefe8ad19ea7569e99f5eb8a803e3785 127.0.0.1:30001 5slots:[0-5460] (5461 slots) master 61 additional replica(s) 7S: dc0702625743c48c75ea935c87813c4060547cef 127.0.0.1:30006 8slots: (0 slots) slave 9 replicates 3da35c40c43b457a113b539259f17e7ed616d13d10M: df0190853a53d8e078205d0e2fa56046f20362a7 127.0.0.1:3000711slots: (0 slots) master12M: f5958382af41d4e1f5b0217c1413fe19f390b55f 127.0.0.1:3000313slots:[10923-16383] (5461 slots) master141 additional replica(s)15M:1d09d26fd755298709efe60278457eaa09cefc26 127.0.0.1:3000816slots: (0 slots) master17M:3da35c40c43b457a113b539259f17e7ed616d13d 127.0.0.1:3000218slots:[5461-10922] (5462 slots) master191 additional replica(s)20S: abec9f98f9c01208ba77346959bc35e8e274b6a3 127.0.0.1:3000521slots: (0 slots) slave22 replicates 887397e6fefe8ad19ea7569e99f5eb8a803e378523S:1a324d828430f61be6eaca7eb2a90728dd5049de 127.0.0.1:3000424slots: (0 slots) slave25 replicates f5958382af41d4e1f5b0217c1413fe19f390b55f26[OK] All nodes agree about slots configuration.27>>> Check for open slots...28>>> Check slots coverage...29[OK] All 16384 slots covered.30[ERR] Node 127.0.0.1:30008 is not empty. Either the node already knows other nodes (check with CLUSTER NODES) or contains some key in database 0.

从以上结果可以看出 30008 节点也被设置成了主节点。

2.添加从节点

使用 cluster replicate nodeId 命令就可以把当前节点设置为目标节点的从节点,执行命令如下:

1127.0.0.1:30008> clusterreplicatedf0190853a53d8e078205d0e2fa56046f20362a72OK 3127.0.0.1:30008> clusternodes4df0190853a53d8e078205d0e2fa56046f20362a7 127.0.0.1:30007@40007 master - 015851478270000 connected 5abec9f98f9c01208ba77346959bc35e8e274b6a3 127.0.0.1:30005@40005 slave 887397e6fefe8ad19ea7569e99f5eb8a803e3785 015851478270001 connected 61a324d828430f61be6eaca7eb2a90728dd5049de 127.0.0.1:30004@40004 slave f5958382af41d4e1f5b0217c1413fe19f390b55f 015851478230003 connected 7887397e6fefe8ad19ea7569e99f5eb8a803e3785 127.0.0.1:30001@40001 master - 015851478260001 connected 0-54608dc0702625743c48c75ea935c87813c4060547cef 127.0.0.1:30006@40006 slave 3da35c40c43b457a113b539259f17e7ed616d13d 015851478269302 connected9f5958382af41d4e1f5b0217c1413fe19f390b55f 127.0.0.1:30003@40003 master - 015851478260003 connected 10923-16383101d09d26fd755298709efe60278457eaa09cefc26 127.0.0.1:30008@40008 myself,slave df0190853a53d8e078205d0e2fa56046f20362a7 015851478230007 connected113da35c40c43b457a113b539259f17e7ed616d13d 127.0.0.1:30002@40002 master - 015851478279332 connected 5461-10922

可以看出 30008 已经变为 30007 的从节点了。

3.删除节点

使用 cluster forget nodeId 命令就可以把一个节点从集群中移除。此命令和 meet 命令不同的时,删除节点需要是使用节点的 Id 进行删除,可以通过 cluster nodes 命令查看所有节点的 Id 信息,其中每一行的最前面的 40 位字母和数组的组合就是该节点的 Id,如下图所示:

惊呆了,竟然可以用这种方式秒建 Redis 集群?

执行命令如下:

1127.0.0.1:30001> clusterforgetdf0190853a53d8e078205d0e2fa56046f20362a72OK

此时我们使用 cluster nodes 命令查看集群的所有节点信息:

1127.0.0.1:30001> clusternodes2dc0702625743c48c75ea935c87813c4060547cef 127.0.0.1:30006@40006 slave 3da35c40c43b457a113b539259f17e7ed616d13d 015851437899406 connected3f5958382af41d4e1f5b0217c1413fe19f390b55f 127.0.0.1:30003@40003 master - 015851437910003 connected 10923-1638343da35c40c43b457a113b539259f17e7ed616d13d 127.0.0.1:30002@40002 master - 015851437890002 connected 5461-109225abec9f98f9c01208ba77346959bc35e8e274b6a3 127.0.0.1:30005@40005 slave 887397e6fefe8ad19ea7569e99f5eb8a803e3785 015851437890005 connected6887397e6fefe8ad19ea7569e99f5eb8a803e3785 127.0.0.1:30001@40001 myself,master - 015851437860001 connected 0-546071a324d828430f61be6eaca7eb2a90728dd5049de 127.0.0.1:30004@40004 slave f5958382af41d4e1f5b0217c1413fe19f390b55f 015851437919454 connected

可以看出之前的端口为 30007 的节点已经被我们成功的移除了。

本文讲了 Redis 集群的两种搭建方式:create-cluster start 和 cluster create,前一种方式虽然速度比较快,但它只能创建数量固定的主从节点,并且所有节点都在同一台服务器上,因此只能用于测试环境。我们还讲了 Redis 集群动态添加主、从节点和删除任意节点的功能。


分享到:


相關文章: