欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

深入淺析Redis 集群伸縮原理

 更新時間:2021年05月15日 09:51:31   作者:LB  
Redis 集群提供了靈活的節(jié)點擴容和收縮方案。在不影響集群對外服務(wù)的情況下,可以為集群添加節(jié)點進行擴容,也可以下線部分節(jié)點進行縮容,接下來通過本文給大家分享Redis 集群伸縮原理,感興趣的朋友一起看看吧

Redis 節(jié)點分別維護自己負責的槽和對應(yīng)的數(shù)據(jù)。伸縮原理:Redis 槽和對應(yīng)數(shù)據(jù)在不同節(jié)點之間移動

環(huán)境:CentOS7 搭建 Redis 集群

一、集群擴容

1. 手動擴容

(1) 準備節(jié)點 9007,并加入集群

192.168.11.40:9001> cluster meet 192.168.11.40 9007

【注意】若 cluster meet 加入已存在于其它集群的節(jié)點,會導致集群合并,造成數(shù)據(jù)錯亂!。建議使用 redis-cli 的 add-node:

# 若節(jié)點已加入其它集群或包含數(shù)據(jù),會報錯
add-node    new_host:new_port existing_host:existing_port
            --cluster-slave  # 直接添加為從節(jié)點
            --cluster-master-id <arg>  # 從節(jié)點對應(yīng)的主節(jié)點id

(2) 遷移槽和數(shù)據(jù)

槽在遷移過程中集群可以正常提供讀寫服務(wù)

首先確定原有節(jié)點的哪些槽需要遷移到新節(jié)點。確保每個節(jié)點負責相似數(shù)量的槽,保證各節(jié)點的數(shù)據(jù)均勻

槽是 Redis 集群管理數(shù)據(jù)的基本單位。數(shù)據(jù)遷移是逐槽進行的

槽遷移流程:

  • 目標節(jié)點準備導入槽的數(shù)據(jù):目標節(jié)點執(zhí)行cluster setslot {slot} importing {sourceNodeId}
  • 源節(jié)點準備遷出槽的數(shù)據(jù):源節(jié)點執(zhí)行cluster setslot {slot} migrating {targetNodeId}
  • 獲取 count 個屬于槽 slot 的鍵:源節(jié)點執(zhí)行cluster getkeysinslot {slot} {count}
  • 遷移鍵:源節(jié)點執(zhí)行migrate {targetIp} {targetPort} "" 0 {timeout} keys {keys...},把鍵通過流水線(pipeline)機制批量遷移到目標節(jié)點。Redis3.0.6 后才支持批量遷移
  • 重復上兩步,直到槽下所有的鍵值數(shù)據(jù)遷移到目標節(jié)點
  • 向集群所有主節(jié)點通知槽被分配給目標節(jié)點:集群內(nèi)所有主節(jié)點執(zhí)行cluster setslot {slot} node {targetNodeId}

內(nèi)部偽代碼:

def move_slot(source,target,slot):
    # 目標節(jié)點準備導入槽
    target.cluster("setslot",slot,"importing",source.nodeId);
    # 目標節(jié)點準備全出槽
    source.cluster("setslot",slot,"migrating",target.nodeId);
    while true :
        # 批量從源節(jié)點獲取鍵
        keys = source.cluster("getkeysinslot",slot,pipeline_size);
        if keys.length == 0:
            # 鍵列表為空時,退出循環(huán)
            break;
        # 批量遷移鍵到目標節(jié)點
        source.call("migrate",target.host,target.port,"",0,timeout,"keys",keys);
        # 向集群所有主節(jié)點通知槽被分配給目標節(jié)點
        for node in nodes:
            if node.flag == "slave":
                continue;
            node.cluster("setslot",slot,"node",target.nodeId);

(3) 將 9001 的槽 4096 遷移到 9007 中

準備數(shù)據(jù)

192.168.11.40:9001> set key:test:5028 value:5028
192.168.11.40:9001> set key:test:68253 value:68253

目標節(jié)點準備工作

192.168.11.40:9007> cluster nodes
8ccdb0963411ebd05ce21952bdd4b7597825afdc 192.168.11.40:9001@19001 master - 0 1620928869000 2 connected 0-5461
bb1bb0f5f9e0ee67846ba8ec94a38da700e2e80d 192.168.11.40:9007@19007 myself,master - 0 1620928868000 0 connected
...
# 9007 準備導入槽 4096 的數(shù)據(jù)
192.168.11.40:9007> cluster setslot 4096 importing 8ccdb0963411ebd05ce21952bdd4b7597825afdc
OK
# 槽 4096 已開啟導入狀態(tài)
192.168.11.40:9007> cluster nodes
bb1bb0f5f9e0ee67846ba8ec94a38da700e2e80d 192.168.11.40:9007@19007 myself,master - 0 1620928959000 0 connected [4096-<-8ccdb0963411ebd05ce21952bdd4b7597825afdc]
...

源節(jié)點準備工作

# 9001 準備導出槽 4096 數(shù)據(jù)
192.168.11.40:9001> cluster setslot 4096 migrating bb1bb0f5f9e0ee67846ba8ec94a38da700e2e80d
OK
# 槽 4096 已開啟導出狀態(tài)
192.168.11.40:9001> cluster nodes
8ccdb0963411ebd05ce21952bdd4b7597825afdc 192.168.11.40:9001@19001 myself,master - 0 1620929179000 2 connected 0-5461 [4096->-bb1bb0f5f9e0ee67846ba8ec94a38da700e2e80d]
...

導出數(shù)據(jù)

# 獲取 100 個屬于槽 4096 的鍵
192.168.11.40:9001> cluster getkeysinslot 4096 100
1) "key:test:5028"
2) "key:test:68253"
# 查看數(shù)據(jù)
192.168.11.40:9001> mget key:test:5028 key:test:68253
1) "value:5028"
2) "value:68253"
# 遷移這2個鍵:migrate 命令保證了每個鍵遷移過程的原子性
192.168.11.40:9001> migrate 192.168.11.40 9007 "" 0 5000 keys key:test:5028 key:test:68253
OK
# 再次查詢會報 ASK 錯誤:引導客戶端找到數(shù)據(jù)所在的節(jié)點
192.168.11.40:9001> mget key:test:5028 key:test:68253
(error) ASK 4096 192.168.11.40:9007

通知所有主節(jié)點:槽 4096 指派給 9007

192.168.11.40:9001> cluster setslot 4096 node bb1bb0f5f9e0ee67846ba8ec94a38da700e2e80d
192.168.11.40:9002> cluster setslot 4096 node bb1bb0f5f9e0ee67846ba8ec94a38da700e2e80d
192.168.11.40:9003> cluster setslot 4096 node bb1bb0f5f9e0ee67846ba8ec94a38da700e2e80d
192.168.11.40:9007> cluster setslot 4096 node bb1bb0f5f9e0ee67846ba8ec94a38da700e2e80d

查看最終結(jié)果

192.168.11.40:9007> cluster nodes
8ccdb0963411ebd05ce21952bdd4b7597825afdc 192.168.11.40:9001@19001 master - 0 1620931743303 7 connected 0-4095 4097-5461
bb1bb0f5f9e0ee67846ba8ec94a38da700e2e80d 192.168.11.40:9007@19007 myself,master - 0 1620931741000 8 connected 4096
...

2. 使用 redis-cli 擴容

redis-cli 提供了槽重分片功能

reshard 命令參數(shù)詳解:

reshard    host:port  # 集群內(nèi)任意節(jié)點地址
           --cluster-from <arg>  # 源節(jié)點id,逗號分隔
           --cluster-to <arg>  # 目標節(jié)點id,只有一個
           --cluster-slots <arg>  # 遷移多少個槽
           --cluster-yes  # 確認執(zhí)行reshard
           --cluster-timeout <arg>  # 每次 migrate 操作的超時時間,默認 60000ms
           --cluster-pipeline <arg>  # 每次批量遷移鍵的數(shù)量,默認 10
           --cluster-replace

將 9001、9002、9003 的槽遷移到 9007,共遷移 4096 個

$ /usr/local/redis/bin/redis-cli --cluster reshard 192.168.11.40:9001
M: 8ccdb0963411ebd05ce21952bdd4b7597825afdc 192.168.11.40:9001
   slots:[0-4095],[4097-5461] (5461 slots) master
   1 additional replica(s)
M: bb1bb0f5f9e0ee67846ba8ec94a38da700e2e80d 192.168.11.40:9007
   slots:[4096] (1 slots) master
...
[OK] All 16384 slots covered.
How many slots do you want to move (from 1 to 16384)? 4096
What is the receiving node ID? bb1bb0f5f9e0ee67846ba8ec94a38da700e2e80d
Please enter all the source node IDs.
  Type 'all' to use all the nodes as source nodes for the hash slots.
  Type 'done' once you entered all the source nodes IDs.
Source node #1: 8ccdb0963411ebd05ce21952bdd4b7597825afdc
Source node #2: 5786e3237c7fa413ed22465d15be721f95e72cfa
Source node #3: 85ceb9826e8aa003169c46fb4ba115c72002d4f9
Source node #4: done
    Moving slot 0 from 8ccdb0963411ebd05ce21952bdd4b7597825afdc
    ...
    Moving slot 12287 from 85ceb9826e8aa003169c46fb4ba115c72002d4f9
Do you want to proceed with the proposed reshard plan (yes/no)? yes
Moving slot 0 from 192.168.11.40:9001 to 192.168.11.40:9007:
...
Moving slot 12287 from 192.168.11.40:9003 to 192.168.11.40:9007:

查看最終結(jié)果

192.168.11.40:9007> cluster nodes
8ccdb0963411ebd05ce21952bdd4b7597825afdc 192.168.11.40:9001@19001 master - 0 1620933907753 7 connected 1366-4095 4097-5461
5786e3237c7fa413ed22465d15be721f95e72cfa 192.168.11.40:9002@19002 master - 0 1620933906733 1 connected 6827-10922
85ceb9826e8aa003169c46fb4ba115c72002d4f9 192.168.11.40:9003@19003 master - 0 1620933905000 3 connected 12288-16383
bb1bb0f5f9e0ee67846ba8ec94a38da700e2e80d 192.168.11.40:9007@19007 myself,master - 0 1620933900000 8 connected 0-1365 4096 5462-6826 10923-12287
...

檢查節(jié)點之間槽的均衡性

$ /usr/local/redis/bin/redis-cli --cluster rebalance 192.168.11.40:9001
...
[OK] All 16384 slots covered.
*** No rebalancing needed! All nodes are within the 2.00% threshold.

遷移之后所有主節(jié)點負責的槽數(shù)量差異在 2% 以內(nèi),因此集群節(jié)點數(shù)據(jù)相對均勻,無需調(diào)整

二、集群收縮

1. 遷移槽

執(zhí)行 reshard 三次,將數(shù)據(jù)平均分布到其他三個節(jié)點

2. 忘記節(jié)點

60s 內(nèi)對所有節(jié)點執(zhí)行如下操作:(不建議)

# 執(zhí)行后,會將該節(jié)點加入禁用列表(持續(xù) 60s),不再向其發(fā)送 Gossip 消息
cluster forget {nodeId}

建議使用 redis-cli 的 del-node 忘記節(jié)點:

/usr/local/redis/bin/redis-cli --cluster del-node {host:port} {nodeId}

內(nèi)部偽代碼

def delnode_cluster_cmd(downNode):
    # 下線節(jié)點不允許包含slots
    if downNode.slots.length != 0
        exit 1
    end
    # 向集群內(nèi)節(jié)點發(fā)送cluster forget
    for n in nodes:
        if n.id == downNode.id:
            # 不能對自己做forget操作
            continue;
        # 如果下線節(jié)點有從節(jié)點則把從節(jié)點指向其他主節(jié)點
        if n.replicate && n.replicate.nodeId == downNode.id :
            # 指向擁有最少從節(jié)點的主節(jié)點
            master = get_master_with_least_replicas();
            n.cluster("replicate",master.nodeId);
        #發(fā)送忘記節(jié)點命令
        n.cluster('forget',downNode.id)
    # 節(jié)點關(guān)閉
    downNode.shutdown();

若主從節(jié)點都要下線,先下線從,避免全量復制

以上就是Redis 集群伸縮原理的詳細內(nèi)容,更多關(guān)于Redis 集群原理的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 使用Ruby腳本部署Redis Cluster集群步驟講解

    使用Ruby腳本部署Redis Cluster集群步驟講解

    今天小編就為大家分享一篇關(guān)于使用Ruby腳本部署Redis Cluster集群步驟講解,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2019-01-01
  • 淺談redis緩存在項目中的使用

    淺談redis緩存在項目中的使用

    最近由于項目需要,在系統(tǒng)緩存服務(wù)部分上用到了redis,本文就淺談下在redis緩存在項目中的使用,感興趣的小伙伴們可以參考一下
    2021-05-05
  • Redisson分布式限流器RRateLimiter的使用及原理小結(jié)

    Redisson分布式限流器RRateLimiter的使用及原理小結(jié)

    本文主要介紹了Redisson分布式限流器RRateLimiter的使用及原理小結(jié),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2024-06-06
  • 如何使用Redis實現(xiàn)電商系統(tǒng)的庫存扣減

    如何使用Redis實現(xiàn)電商系統(tǒng)的庫存扣減

    在日常開發(fā)中有很多地方都有類似扣減庫存的操作,本文主要介紹了如何使用Redis實現(xiàn)電商系統(tǒng)的庫存扣減,具有一定的參考價值,感興趣的可以了解一下
    2022-01-01
  • 關(guān)于redigo中PubSub的一點小坑分析

    關(guān)于redigo中PubSub的一點小坑分析

    這篇文章主要給大家介紹了關(guān)于redigo中PubSub的一點小坑的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-01-01
  • redis生成全局id的實現(xiàn)步驟

    redis生成全局id的實現(xiàn)步驟

    生成全局唯一的標識符是非常常見的需求,本文主要介紹了redis生成全局id的實現(xiàn)步驟,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2024-05-05
  • Redis集群模式和常用數(shù)據(jù)結(jié)構(gòu)詳解

    Redis集群模式和常用數(shù)據(jù)結(jié)構(gòu)詳解

    Redis集群模式下的運維指令主要用于集群的搭建、管理、監(jiān)控和維護,講解了一些常用的Redis集群運維指令,本文重點介紹了Redis集群模式和常用數(shù)據(jù)結(jié)構(gòu),需要的朋友可以參考下
    2024-03-03
  • redis實現(xiàn)分布式session的解決方案

    redis實現(xiàn)分布式session的解決方案

    session存放在服務(wù)器,關(guān)閉瀏覽器不會失效,本文主要介紹了redis實現(xiàn)分布式session的解決方案,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • Redis如何優(yōu)雅的刪除特定前綴key

    Redis如何優(yōu)雅的刪除特定前綴key

    這篇文章主要給大家介紹了關(guān)于Redis如何優(yōu)雅的刪除特定前綴key的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家學習或者使用Redis具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧
    2019-06-06
  • redis cluster支持pipeline的實現(xiàn)思路

    redis cluster支持pipeline的實現(xiàn)思路

    本文給大家介紹redis cluster支持pipeline的實現(xiàn)思路,在 cluster 上執(zhí)行 pipeline 可能會由于 redis 節(jié)點擴縮容 中途 redirection 切換連接導致結(jié)果丟失,具體細節(jié)問題請參考下本文
    2021-06-06

最新評論