Redis 集群演進探討和總結

来源:https://www.cnblogs.com/svenaugustus/archive/2020/05/22/12938653.html
-Advertisement-
Play Games

Redis為什麼需要集群? 首先Redis單實例主要有單點,容量有限,流量壓力上限的問題。 Redis單點故障,可以通過主從複製 ,和自動故障轉移 哨兵機制。 但Redis單 實例提供讀寫服務,仍然有容量和壓力問題,因此需要數據分區,構建多個 實例同時提供讀寫服務(不僅限於從 節點提供讀服務)。 那 ...


Redis為什麼需要集群?

首先Redis單實例主要有單點,容量有限,流量壓力上限的問題。

Redis單點故障,可以通過主從複製replication,和自動故障轉移sentinel哨兵機制。

但Redis單Master實例提供讀寫服務,仍然有容量和壓力問題,因此需要數據分區,構建多個Master實例同時提供讀寫服務(不僅限於從replica節點提供讀服務)。

那麼就需要一定的機制保證數據分區。這樣能充分把容量分攤到多台電腦,或能充分利用多核電腦的性能。

並且數據在各個主Master節點間不能混亂,當然最好還能支持線上數據熱遷移的特性。

探討數據分區方案

file
針對數據分區,一般來說,分為兩個大類:

  • 邏輯拆分:
    邏輯上能拆分,比如 Redis 中的 M1 節點 存儲 A服務需要的業務數據,而 Redis 中的 M2 節點存儲 B服務需要的業務數據。
  • 數據分區:
    當邏輯上不能拆分,那麼只能按數據來拆分,需要保證客戶端讀和寫數據一致。
    因此需要一個高效快速的數據結構來路由對應的Master節點。
    最容易想到的就是類比 Java 中的 HashMap, 採用 哈希演算法,快速找到,快速設置。
    這裡有四種方式,分別是固定取模,隨機,哈希一致性,哈希槽。

固定取模

file
假設有三個 Master,配置IP 和權重如下:

Real Server IP weight
10.0.2.21 1
10.0.2.22 2
10.0.2.23 3

那麼會根據每一個real Server 及其權重虛擬出對應權重 weight 個的虛擬vritual server節點,映射關係會是:

Real Server IP virtual server
10.0.2.21 1
10.0.2.22 2,3
10.0.2.23 4,5,6

一個 key 存儲在那個虛擬vritual server節點,通過哈希hash演算法:

virtual_server_index = hash(key) % (total_virtual_weight)

假設某個key,它的 hash 值是 10,那麼以上: 10%6=4,將落到 10.0.2.23 這個真實的 Master上。

  • 缺點
    因為取模的模數是固定的,當新增或刪除 master節點時,所有的數據幾乎要全部洗牌,幾乎需要重新遷移數據(而且相當麻煩),無法做到線上數據熱遷移。
    意味著Redis在此種用法下,只能當緩存,不能當存儲資料庫!

隨機

file
隨機選取一個存儲和訪問。
一般結合 list,用於非順序性要求的消息隊列場景。

  • 缺點:
    使用場景比較單一。
    並且由於隨機性問題,導致持久化存在不可靠性。Redis在此種用法下,也只能當緩存,不能當存儲資料庫!

一致性哈希

一致性哈希演算法(Consistent Hashing)最早在論文《Consistent Hashing and Random Trees: Distributed Caching Protocols for Relieving Hot Spots on the World Wide Web》中被提出。
簡單來說,一致性哈希將整個哈希值空間組織成一個虛擬的圓環,如假設某哈希函數H的值空間為0-2^32-1(即哈希值是一個32位無符號整形),整個哈希空間環如下:
file

  • 1.有一個HASH環,環上每個節點都是一個自然數,從0開始順時針遞增,直到2^32-1,最後回到0

  • 2.真實節點 M1 M2 M3 通過 hash(IP 或主機名)確定在哈希環上的位置

  • 3.當客戶端請求時,首先 hash(key) 確定在哈希環上的位置,然後順時針往後找,找到的第一個真實節點,就是客戶端需要請求訪問的真實主機

  • 優點:
    哈希一致性其實是對固定取模的一種優化。
    (1)擴展性:當增加節點時,只會影響順時針的真實節點(此部分數據比較難遷移),而不是影響全部的節點。
    (2)容錯性:當節點宕機或刪除節點時,只會影響逆時針的真實節點,而不是影響全部的節點。
    (3)平衡性:當哈希演算法的節點過少時,會可能造成某些伺服器的數據存儲較多,而另外一些存儲較少,造成數據傾斜,當節點足夠多時,這種現象得以緩解。
    因此虛擬節點個數較大的時候,數據的平衡性得以保證。

  • 缺點:
    因為當增刪節點時,需要重新計算受影響部分的節點中的key全部找出來,才能遷移,這個很麻煩!!!
    Redis在此種用法下,也只能當緩存,不能當存儲資料庫!

哈希槽(PreSharding,預先分片)

這個跟哈希一致性很相似。
區別在於,它預先分配好真實節點管理的哈希槽(slot),並存儲管理起來,我們可以預先知道哪個master主機擁有哪些哈希槽(slot),這裡總數是16384。
file

127.0.0.1:7001> cluster nodes
2aaf59558f1b9f493a946a695e51711eb03d15f9 127.0.0.1:7002@17002 master - 0 1590126183862 2 connected 5461-10922
6439c3e9468fd2c545a63b3b9bfe658c5fc14287 127.0.0.1:7003@17003 master - 0 1590126181856 3 connected 10923-16383
340d985880c23de9816226dff5fd903322e44313 127.0.0.1:7001@17001 myself,master - 0 1590126182000 1 connected 0-5460

我們可以清晰看到Redis Cluster中的每一個master節點管理的哈希槽。
比如 127.0.0.1:7001 擁有哈希槽 0-5460, 127.0.0.1:7002 擁有哈希槽 5461-10922, 127.0.0.1:7003 擁有哈希槽 10923-16383。
file

➜  redis-cli -p 7001         
127.0.0.1:7001> set a 1
(error) MOVED 15495 127.0.0.1:7003

➜  redis-cli -p 7001 -c
127.0.0.1:7001> set a 1
-> Redirected to slot [15495] located at 127.0.0.1:7003
OK

我們看到的是master節點在 Redis Cluster中的實現時,都存有所有的路由信息。
當客戶端的key 經過hash運算,發送slot 槽位不在本節點的時候。
(1)如果是非集群方式連接,則直接報告錯誤給client,告訴它應該訪問集群中那個IP的master主機。
(2)如果是集群方式連接,則將客戶端重定向到正確的節點上。
註意這裡並不是127.0.0.1:7001 幫client去連接127.0.0.1:7003獲取數據的,而是將客戶端請求重定向了。

  • 優點:
    繼承並增強一致性哈希的容錯性,擴展性,以及平衡性。
    Redis在此種用法下,可以當緩存,也能當存儲資料庫!

  • 這裡Redis給出更詳細的說明:https://redis.io/topics/partitioning

具體方案

以下列表為按照出現的先後順序排列:

方案 描述 數據分區支持策略 分散式 線上數據熱遷移
twemproxy twitter 開源的redis代理中間件,不修改redis源碼 https://github.com/twitter/twemproxy 存在modula(固定取模)、 random (隨機)、ketama(哈希一致性)三種可選的配置 本身是單點的,可以通過keepalived等保證高可用 不支持,無法平滑地擴容/縮容
Redis Cluster 官方提供的集群方案 採用預先分片(PreSharding),即哈希槽方式,存儲在每一個master節點上 沒有proxy代理層,客戶端可以連接集群中的任意master節點 提供客戶端命令redis-cli --cluster reshard ip port按哈希槽遷移指定節點的數據
codis 豌豆莢開源的redis代理中間件,修改了redis源碼 https://github.com/CodisLabs/codis 採用預先分片(PreSharding),即哈希槽方式,存儲在ZooKeeper上 集群部署,部署相對複雜 支持數據熱遷移

@SvenAugustus(https://www.flysium.xyz/)
更多請關註微信公眾號【編程不離宗】,專註於分享伺服器開發與編程相關的技術乾貨:


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • STM32 HAL庫 +freeRTOS+Keil 移植 官方freeRTOS移植教程很多,本文不做贅述。本文基於Keil 5提供的freeRTOS庫,進行移植。 Keil 版本:Keil MDK uVision5 首先準備好一個可以用的HAL庫工程,為了方便調試,選擇了原子哥的開發板上提供跑馬燈工 ...
  • 一、用戶操作1.增加用戶 useradd [username] 2.設置用戶密碼 passwd [username] 3.刪除用戶 userdel [username] 4.查看所有用戶 cat /etc/passwd 5.查看當前活躍用戶 w 6.查看簡明用戶列表cat /etc/passwd|g ...
  • 功能描述:切換目錄 語法:cd [目標目錄] “[ ]”為可選項。 說明:cd命令可以讓用戶在不同的目錄間來回切換,不過該用戶需要有足夠的許可權才能進入目標目錄。 示例1:從當前目錄切換至/tmp目錄 cd :不帶任何參數時表示切換回用戶的家目錄。 cd \~ :切換回自己的家目錄。註意:在bash中 ...
  • [toc] route命令 功能說明:顯示或管理路由表,路由表是在內核中的,route命令會立刻修改內核中的路由表,立即生效,系統重啟後失效。 用法 route [ n] route add [ net| host] target [netmask Nm] [gw Gw] [[dev] If] ro ...
  • 下載Redis安裝包並解壓 在 /opt/soft 內下載 "redis 5.0.5" 解壓完之後, /opt/module/ ⽬錄中會出現⼀個 的⽬錄 編譯並安裝 將Redis 安裝為系統服務並後臺啟動 設置允許遠程連接與訪問密碼 檢驗安裝結果 ...
  • 最近談到Redis就會聽到哨兵模式,工作期間同事也分享過關於哨兵模式的知識,但由於工作忙(給自己找個藉口)沒有沒認真看,現在惡補下,老樣子還是分上篇應用,下篇看實現過程,下麵我們來看下哨兵到底是啥? 哨兵模式(Sentinel)是Redis的高可用解決方案。由一個或多個Sentinel實例組成的Se ...
  • 檢測是否存在MARIADB 如果系統之前⾃帶 ,可以先卸載之。 ⾸先查詢已安裝的 Mariadb 安裝包: 卸載mariadb 刪除配置文件,刪除數據目錄 下載MYSQL安裝包並解壓 在 /opt/soft 內下載 "MySQL 5.7" 解壓完之後, /opt/module/ ⽬錄中會出現⼀個 的 ...
  • INSERT INTO test_table_public(class, name, geography) SELECT class, name, geography FROM test_table WHERE id >= 137181 AND id <= 137214; SELECT class, ...
一周排行
    -Advertisement-
    Play Games
  • 概述:在C#中,++i和i++都是自增運算符,其中++i先增加值再返回,而i++先返回值再增加。應用場景根據需求選擇,首碼適合先增後用,尾碼適合先用後增。詳細示例提供清晰的代碼演示這兩者的操作時機和實際應用。 在C#中,++i 和 i++ 都是自增運算符,但它們在操作上有細微的差異,主要體現在操作的 ...
  • 上次發佈了:Taurus.MVC 性能壓力測試(ap 壓測 和 linux 下wrk 壓測):.NET Core 版本,今天計劃準備壓測一下 .NET 版本,來測試並記錄一下 Taurus.MVC 框架在 .NET 版本的性能,以便後續持續優化改進。 為了方便對比,本文章的電腦環境和測試思路,儘量和... ...
  • .NET WebAPI作為一種構建RESTful服務的強大工具,為開發者提供了便捷的方式來定義、處理HTTP請求並返迴響應。在設計API介面時,正確地接收和解析客戶端發送的數據至關重要。.NET WebAPI提供了一系列特性,如[FromRoute]、[FromQuery]和[FromBody],用 ...
  • 原因:我之所以想做這個項目,是因為在之前查找關於C#/WPF相關資料時,我發現講解圖像濾鏡的資源非常稀缺。此外,我註意到許多現有的開源庫主要基於CPU進行圖像渲染。這種方式在處理大量圖像時,會導致CPU的渲染負擔過重。因此,我將在下文中介紹如何通過GPU渲染來有效實現圖像的各種濾鏡效果。 生成的效果 ...
  • 引言 上一章我們介紹了在xUnit單元測試中用xUnit.DependencyInject來使用依賴註入,上一章我們的Sample.Repository倉儲層有一個批量註入的介面沒有做單元測試,今天用這個示例來演示一下如何用Bogus創建模擬數據 ,和 EFCore 的種子數據生成 Bogus 的優 ...
  • 一、前言 在自己的項目中,涉及到實時心率曲線的繪製,項目上的曲線繪製,一般很難找到能直接用的第三方庫,而且有些還是定製化的功能,所以還是自己繪製比較方便。很多人一聽到自己畫就害怕,感覺很難,今天就分享一個完整的實時心率數據繪製心率曲線圖的例子;之前的博客也分享給DrawingVisual繪製曲線的方 ...
  • 如果你在自定義的 Main 方法中直接使用 App 類並啟動應用程式,但發現 App.xaml 中定義的資源沒有被正確載入,那麼問題可能在於如何正確配置 App.xaml 與你的 App 類的交互。 確保 App.xaml 文件中的 x:Class 屬性正確指向你的 App 類。這樣,當你創建 Ap ...
  • 一:背景 1. 講故事 上個月有個朋友在微信上找到我,說他們的軟體在客戶那邊隔幾天就要崩潰一次,一直都沒有找到原因,讓我幫忙看下怎麼回事,確實工控類的軟體環境複雜難搞,朋友手上有一個崩潰的dump,剛好丟給我來分析一下。 二:WinDbg分析 1. 程式為什麼會崩潰 windbg 有一個厲害之處在於 ...
  • 前言 .NET生態中有許多依賴註入容器。在大多數情況下,微軟提供的內置容器在易用性和性能方面都非常優秀。外加ASP.NET Core預設使用內置容器,使用很方便。 但是筆者在使用中一直有一個頭疼的問題:服務工廠無法提供請求的服務類型相關的信息。這在一般情況下並沒有影響,但是內置容器支持註冊開放泛型服 ...
  • 一、前言 在項目開發過程中,DataGrid是經常使用到的一個數據展示控制項,而通常表格的最後一列是作為操作列存在,比如會有編輯、刪除等功能按鈕。但WPF的原始DataGrid中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...