瞭解一致性哈希演算法

来源:https://www.cnblogs.com/myzony/archive/2019/03/26/10601554.html
-Advertisement-
Play Games

一、背景 1.1 使用場景 一致性哈希演算法一般用於解決分散式系統當中的熱點問題,用於提升分散式系統的可擴展性與健壯性。 1.2 解決的問題 一般用於分散式緩存系統當中的緩存擊穿問題,簡單哈希在服務節點數量產生變化的時候,其緩存命中率很低,從而導致大量介面直接請求資料庫,造成緩存擊穿的情況。 例如我們 ...


一、背景

1.1 使用場景

一致性哈希演算法一般用於解決分散式系統當中的熱點問題,用於提升分散式系統的可擴展性與健壯性。

1.2 解決的問題

一般用於分散式緩存系統當中的緩存擊穿問題,簡單哈希在服務節點數量產生變化的時候,其緩存命中率很低,從而導致大量介面直接請求資料庫,造成緩存擊穿的情況。

例如我們有 10 台緩存伺服器,我們可以對請求關鍵字 (key) 進行哈希操作,將其值對 10 取模,得到的結果即伺服器的索引。

伺服器信息=hash(key) % 10

但是一旦增加/減少了一臺伺服器,其所有緩存數據的位置都會發生相應的改變。例如原本 Id 為 2 的用戶信息,取模的結果為 hash(2)%10=4 ,當增加了一臺伺服器之後 hash(2)%11=? ,這裡的緩存位置就被改變了,這個時候就需要一致性哈希來解決問題了。

一致性哈希可以解決的問題:

  1. 提高緩存命中率。
  2. 緩存數據應該均勻地分配在各個伺服器中。

二、原理

註意:

由於粗心,將伺服器 C 的 IP 地址也設置成了 192.168.100.102,其 IP 應該是 192.168.100.103。

  1. 創建一個環,這個哈希環有 2^32 個節點。

  2. 求出伺服器的哈希值,並將其與哈希環的節點數量取模,根據得到的位置,把服務分配在一個哈希環當中。

  3. 根據存儲數據的鍵,求出其哈希值,也將其映射在哈希環上。

  4. 根據鍵在哈希環的位置,順時針查找第一個服務節點,將數據存儲到對應的伺服器當中。

  5. 如果增加了一臺新的伺服器 D,僅會影響 D 之前的區間數據。

  6. 當我們需要獲得某個緩存數據在哪個伺服器的時候,僅需要對這個數據的關鍵字取其 Hash 值,並對 2^32 取模,找到它下一個節點即是數據所對應的伺服器節點。

2.1 新的問題

使用上述方案的確可以解決由於服務 節點增加/減少 造成的緩存擊穿,這是節點分佈位置均衡的情況。如果節點的在哈希環上 分佈位置不均勻 ,就會造成下圖這種極端情況。

上圖中,黃色的小點代表緩存的數據,A、B、C 並沒有均勻地分佈在哈希環上。可以看到 C -> A 區間是最大的,這就會造成大部分數據都會存放到 A 節點當中,導致 伺服器雪崩 的情況發生。

2.2 使用虛擬節點解決問題

由於伺服器節點可能存在分佈不均的問題,我們可以將一些虛擬節點放在哈希環上,這些虛擬節點其實是 映射 的真實伺服器的地址。

從下圖來看,黑底白字的伺服器節點 A、B、C 只是真實節點的三個副本,虛擬節點 B -> C 區間的內容,實際上是存放到真實 C 節點的。我們就可以通過增加虛擬節點來解決伺服器節點分佈不均的問題。

三、實現

這裡的 DEMO 我們使用的是 C# + .NET Core 進行實現,在這個 DEMO 當中演示了基於一致性哈希演算法的緩存的

using System;
using System.Collections.Generic;
using System.Security.Cryptography;
using System.Text;

/*
 * 一致性哈希演算法的 DEMO,主要用於演示一致性哈希演算法的實現與實際應用。
 */

namespace ConsistentHashing.Startup
{
    public class NodeInfo
    {
        public string IPAddress { get; set; }
    }

    public class VirtualNodeInfo
    {
        public string NodeName { get; set; }

        public NodeInfo RealNodeInfo { get; set; }
    }

    public class ConsistentHashing
    {
        // 哈希環的大小
        private readonly int _ringCount;
        // 每個物理節點對應的虛擬節點數量
        private readonly int _virtualNodeCount;

        // 哈希環
        public readonly VirtualNodeInfo[] HashRing;

        public ConsistentHashing(int ringCount = int.MaxValue, int virtualNodeCount = 3)
        {
            _ringCount = ringCount;
            _virtualNodeCount = virtualNodeCount;
            HashRing = new VirtualNodeInfo[_ringCount];
        }

        public NodeInfo GetNode(string key)
        {
            var pos = Math.Abs(GetStandardHashCode(key) % _ringCount);
            // 順時針查找最近的節點
            return GetFirstNodeInfo(pos).RealNodeInfo;
        }

        /// <summary>
        /// 向哈希環上添加虛擬節點。
        /// </summary>
        public void AddNode(NodeInfo info)
        {
            for (int index = 0; index < _virtualNodeCount; index++)
            {
                // 哈希環上沒有物理節點,只有虛擬節點
                var virtualNodeName = $"{info.IPAddress}#{index}";
                var hashIndex = Math.Abs(GetStandardHashCode(virtualNodeName) % _ringCount);
                
                // 搜索空的哈希環位置
                var emptyIndex = GetEmptyNodeIndex(hashIndex);

                if (emptyIndex == -1)
                {
                    break;
                }
                
                HashRing[emptyIndex] = new VirtualNodeInfo{NodeName = virtualNodeName,RealNodeInfo = info};
            }
        }

        public void RemoveNode(NodeInfo info)
        {
            // 構建虛擬節點的 KEY
            var virtualNames = new List<string>();
            for (int i = 0; i < _virtualNodeCount; i++)
            {
                virtualNames.Add($"{info.IPAddress}#{i}");
            }

            for (int i = 0; i < HashRing.Length; i++)
            {
                if(HashRing[i] == null) continue;
                if (virtualNames.Contains(HashRing[i].NodeName)) HashRing[i] = null;
            }
        }

        /// <summary>
        /// 計算指定 KEY 的 HASH 值
        /// </summary>
        private int GetStandardHashCode(string key)
        {
            var sha1 = SHA256.Create();
            var hashValue = sha1.ComputeHash(Encoding.UTF8.GetBytes(key));
            return BitConverter.ToInt32(hashValue);
        }

        /// <summary>
        /// 迴圈遍歷哈希環,尋找空節點的索引,防止覆蓋存在的節點信息。
        /// </summary>
        private int GetEmptyNodeIndex(int startFindIndex)
        {
            while (true)
            {
                if (HashRing[startFindIndex] == null)
                {
                    return startFindIndex;
                }

                var nextIndex = GetNextNodeIndex(startFindIndex);
                // 說明已經遍歷了整個哈希環,說明沒有空的節點了。
                if (startFindIndex == nextIndex)
                {
                    return -1;
                }

                startFindIndex = nextIndex;
            }
        }

        /// <summary>
        /// 根據指定的索引,獲得哈希環的下一個索引。這裡需要註意的是,因為哈希環是一個環形,當
        /// 當前位置為環的末尾時,應該從 0 開始查找。
        /// </summary>
        private int GetNextNodeIndex(int preIndex)
        {
            if (preIndex == HashRing.Length - 1) return 0;

            return ++preIndex;
        }

        private VirtualNodeInfo GetFirstNodeInfo(int currentIndex)
        {
            VirtualNodeInfo nodeInfo = null;
            int nextIndex = currentIndex;
            while (nodeInfo == null)
            {
                nodeInfo = HashRing[GetNextNodeIndex(nextIndex)];
                nextIndex += 1;
            }
            
            return nodeInfo;
        }
    }

    internal class Program
    {
        private static void Main(string[] args)
        {
            var consistentHashing = new ConsistentHashing(400,10);
            consistentHashing.AddNode(new NodeInfo {IPAddress = "192.168.1.101"});
            consistentHashing.AddNode(new NodeInfo {IPAddress = "192.168.1.102"});
            consistentHashing.AddNode(new NodeInfo {IPAddress = "192.168.1.103"});
            consistentHashing.AddNode(new NodeInfo {IPAddress = "192.168.1.104"});

            foreach (var node in consistentHashing.HashRing)
            {
                Console.WriteLine(node?.NodeName ?? "NULL");
            }

            // 存放 Id 為 15 的緩存伺服器
            var nodeInfo = consistentHashing.GetNode("15");
            
            // 刪除節點測試
            consistentHashing.RemoveNode(new NodeInfo {IPAddress = "192.168.1.103"});
        }
    }
}

以上 DEMO 非線程安全,在實際項目使用當中應該考慮線程同步的問題。


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

-Advertisement-
Play Games
更多相關文章
  • socket基礎 什麼是socket? - socket為介面通道,內部封裝了IP地址、埠、協議等信息;我們可以看作是以前的通過電話機撥號上網的年代,socket即為電話線 socket通信流程 我們通過下麵的圖來瞭解socket的通信流程 流程描述: - 1 伺服器根據地址類型(ipv4,ipv ...
  • SQLAlchemy 是一種 ORM 框架,通過使用它,可以大大簡化我們對資料庫的操作,不用再寫各種複雜的 了。 說明 操作系統:Windows 10 Python 版本:3.7x 虛擬環境管理器:virtualenv 代碼編輯器:VS Code 實驗目標 實現網站與 mysql 資料庫的連接和 C ...
  • 1. 前言 最近重溫了《Framework Design Guidelines》。 《Framework Design Guidelines》中文名稱為《.NET設計規範 約定、慣用法與模式》,簡介如下: 數千名微軟精銳開發人員的經驗和智慧,最終濃縮在這本設計規範之中。與上一版相比,書中新增了許多評 ...
  • //用C#自帶的壓縮,最少要.net4.5或以上,先增加引用 System.IO.Compression.FileSystem // FolderBrowserDialog dlg = new FolderBrowserDialog(); //壓縮目錄 顯示一個標準選擇文件夾對話框 OpenFile ...
  • 原文地址:http://www.entityframeworktutorial.net/EntityFramework-Architecture.aspx 下麵的圖形,展示了EF的總體架構: 讓我們來分別看看,每個組件都是啥吧: EDM(Entity Data Model)【實體數據模型】:EDM( ...
  • 今年來了新公司,公司沒有用什麼新技術,架構就簡單的前後分離,但是我推一下新的技術,在這基礎上我要培訓一下同事,讓他們能接受,對新技術不感到陌生,並且認可願意去學習。其實在這個過程中也能讓他們認同我這個人吧!老闆是一位曾經在9幾年寫過一段時間代碼的人,對新的技術什麼的不是很瞭解,我提的建議什麼的很難去 ...
  • 最近在做項目的時候遇到一種情:用C#程式以管理員許可權去執行一個bat文件,且此bat文件裡面有cd命令來進入文件的下一級目錄,比如: echo test begin cd test1 setup1.exe cd test2 setup2.exe echo test finished echo off ...
  • 規劃、梳理、構建電子化的流程系統平臺 梳理成企業的行政類、人事類、項目類、財務類。所有的流程都是企業管理制度的電子化表達,通過固化流程系統的構建支撐企業管理控制行落地 建立全新公司統一的工作流管理平臺,採用電子化的流程,突破各種邊界,進行胯部門,跨企業的及時溝通,構造協作的環境。系統支持自定義各種簡 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...