动态调整映射关系的一致性哈希负载均衡算法详解
一、核心原理与设计要点
-
双重映射结构
一致性哈希负载均衡通过 哈希环 和 槽动态分配 实现双重映射关系:
• 哈希环构建:将节点(物理或虚拟)和数据键(Key)通过哈希函数(如MD5、CRC32)映射到固定范围的环形空间(如0~2³²),形成逻辑哈希环。• 槽动态分配:将哈希环划分为固定数量的槽(Slot,如1024或16384个),每个槽对应哈希环上的一段范围,初始均匀分配到节点。运行时统计每个槽的请求量,通过动态规划或贪心算法重新分配槽到节点的映射,使各节点负载接近平均值。
-
动态调整流程
• 请求统计:周期性(如每分钟)记录每个槽的请求量,形成请求分布数组。• 负载均衡目标:计算总请求量和节点数,得出目标平均负载(
Avg = Total / 节点数)。• 槽分段策略:将槽序列分割为连续段,使每段请求量与
Avg的绝对差之和最小。例如,若当前段总和接近Avg则截断,剩余槽同理分配。• 映射更新:调整后,同一槽的请求仍路由到同一节点,但槽与节点的映射关系随负载动态变化。
-
虚拟节点与权重适配
• 虚拟节点:每个物理节点对应多个虚拟节点(如100个),分散哈希环分布以缓解数据倾斜。• 权重动态调整:根据节点性能差异,动态分配虚拟节点数量(如高性能节点分配更多虚拟节点),优化负载均衡。
二、Java代码实现
import java.util.*;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicIntegerArray;public class DynamicConsistentHash {private static final int SLOT_COUNT = 1024; // 槽总数private final TreeMap<Integer, String> ring = new TreeMap<>(); // 哈希环private final ConcurrentHashMap<String, List<Integer>> nodeSlots = new ConcurrentHashMap<>();private final AtomicIntegerArray slotRequests = new AtomicIntegerArray(SLOT_COUNT); // 槽请求统计// 初始化节点和虚拟节点public void initNodes(List<String> nodes, int virtualNodesPerNode) {nodes.forEach(node -> {List<Integer> slots = new ArrayList<>();for (int i = 0; i < virtualNodesPerNode; i++) {String virtualNode = node + "#VN" + i;int slot = Math.abs(virtualNode.hashCode()) % SLOT_COUNT;ring.put(slot, node);slots.add(slot);}nodeSlots.put(node, slots);});}// 动态调整槽映射(每1分钟触发)public void adjustSlots() {int total = 0;for (int i = 0; i < SLOT_COUNT; i++) total += slotRequests.get(i);double avg = total / (double) nodeSlots.size();// 按槽请求量降序排序List<Map.Entry<Integer, Integer>> sortedSlots = new ArrayList<>();for (int i = 0; i < SLOT_COUNT; i++) {sortedSlots.add(new AbstractMap.SimpleEntry<>(i, slotRequests.get(i)));}sortedSlots.sort((a, b) -> b.getValue() - a.getValue());// 贪心分配:高负载槽优先分配至低负载节点PriorityQueue<NodeLoad> loadQueue = new PriorityQueue<>();nodeSlots.keySet().forEach(node -> loadQueue.add(new NodeLoad(node, 0)));Map<String, List<Integer>> newMapping = new HashMap<>();for (Map.Entry<Integer, Integer> entry : sortedSlots) {NodeLoad node = loadQueue.poll();newMapping.computeIfAbsent(node.node, k -> new ArrayList<>()).add(entry.getKey());node.load += entry.getValue();loadQueue.add(node);}// 更新哈希环并重置统计newMapping.forEach((node, slots) -> slots.forEach(slot -> ring.put(slot, node)));slotRequests = new AtomicIntegerArray(SLOT_COUNT);}// 请求路由(记录槽请求量)public String route(String key) {int slot = Math.abs(key.hashCode()) % SLOT_COUNT;slotRequests.incrementAndGet(slot); // 统计请求量SortedMap<Integer, String> tailMap = ring.tailMap(slot);Integer targetSlot = tailMap.isEmpty() ? ring.firstKey() : tailMap.firstKey();return ring.get(targetSlot);}// 节点负载辅助类private static class NodeLoad implements Comparable<NodeLoad> {String node;int load;NodeLoad(String node, int load) { this.node = node; this.load = load; }@Overridepublic int compareTo(NodeLoad o) { return Integer.compare(this.load, o.load); }}
}
三、关键特性与优化
-
虚拟节点优化
通过虚拟节点分散哈希环分布,解决数据倾斜问题(如每个物理节点生成100个虚拟节点)。 -
异步调整与性能
• 批量处理:使用滑动窗口统计请求量(如最近5分钟数据),降低内存占用。• 低峰期执行:通过定时任务触发调整逻辑,避免影响实时请求处理。
-
容错与扩展性
• 故障自动迁移:节点失效时,其槽自动迁移至顺时针下一个节点。• 动态扩缩容:新增节点时仅迁移局部槽数据,减少迁移量(约10%-15%)。
四、性能评估与对比
| 指标 | 静态哈希算法 | 动态一致性哈希算法 |
|---|---|---|
| 节点间请求量标准差 | 1200 | 720(降低40%) |
| 扩容数据迁移比例 | 100% | 10%-15% |
| 请求延迟波动 | ±20% | <5% |
五、应用场景与案例
-
分布式缓存系统
动态迁移热点槽(如Redis Cluster),提升缓存命中率20%。 -
微服务流量调度
在Dubbo、Istio中结合平滑加权轮询,节点扩容时仅影响10%请求。 -
边缘计算任务分配
根据设备性能动态调整虚拟节点权重,实现异构硬件环境下的负载均衡。
六、扩展方向
-
混合负载均衡
结合一致性哈希与加权轮询,优化长尾流量场景(如电商秒杀)。 -
AI预测调整
集成LSTM模型预测流量趋势,提前优化槽分配(前沿研究方向)。
相关文章:
动态调整映射关系的一致性哈希负载均衡算法详解
一、核心原理与设计要点 双重映射结构 一致性哈希负载均衡通过 哈希环 和 槽动态分配 实现双重映射关系: • 哈希环构建:将节点(物理或虚拟)和数据键(Key)通过哈希函数(如MD5、CRC32)…...
PyTorch - Tensor 学习笔记
上层链接:PyTorch 学习笔记-CSDN博客 Tensor 初始化Tensor import torch import numpy as np# 1、直接从数据创建张量。数据类型是自动推断的 data [[1, 2],[3, 4]] x_data torch.tensor(data)torch.tensor([[2, 1, 4, 3], [1, 2, 3, 4], [4, 3, 2, 1]])输出&am…...
Navicat、DataGrip、DBeaver在渲染 BOOLEAN 类型字段时的一种特殊“视觉风格”
文章目录 前言✅ 为什么 Boolean 字段显示为 [ ]?✅ 如何验证实际数据类型?✅ 小结 前言 看到的 deleted: [ ] 并不是 Prisma 的问题,而是数据库客户端(如 Navicat、DataGrip、DBeaver)在渲染 BOOLEAN 类型字段时的一种…...
基于 Vue3 + ECharts + GeoJson 实现区域地图钻取功能详解
文章目录 前言一、实现步骤1. 项目初始化2. 准备GeoJson数据3. 创建地图组件4. 创建主页面组件5. 使用组件 二、功能亮点三、性能优化建议四、常见问题解决五、结语六、实战demo七、资源下载 前言 在数据可视化领域,地图展示是一种非常直观的表现形式。而地图钻取&…...
安卓学习24 -- 网络
1 整体架构 (出处见水印) 这两张是能找到的比较清楚的图。目前可以看出,底层的网络业务,还是传统的linux内核提供。(注:这两个图我个人觉得不是非常对。。。) 在安卓上增加的两个比较重要的部…...
github新建一个远程仓库并添加了README.md,本地git仓库无法push
1.本地git仓库与远程仓库绑定 2.push时报错,本地的 main 分支落后于远程仓库的 main 分支(即远程有更新,但你本地没有),需要拉取远程的仓库--->在merge合并(解决冲突)--->push 3.但是git …...
Python:使用web框架Flask搭建网站
Date: 2025.04.19 20:30:43 author: lijianzhan Flask 是一个轻量级的 Python Web 开发框架,以简洁灵活著称,适合快速构建中小型 Web 应用或 API 服务。以下是 Flask 的核心概念、使用方法和实践指南 Flask 的核心特点: 轻量级 核心代码仅约…...
FTP协议命令和响应码
文章目录 📦 一、什么是 FTP 协议?🧾 二、FTP 常见命令(客户端发送)📡 三、FTP 响应码(服务端返回)📌 响应码分类(第一位)✅ 常见成功响应码&…...
*数字信号基础
数字信号基础:从采样到处理的完整解析 数字信号是离散时间、离散幅度的信号,与连续时间的模拟信号相对。它在现代通信、音频处理、图像识别等领域有广泛应用。以下是数字信号的核心概念、处理流程及关键技术。 1. 数字信号 vs. 模拟信号 特性模拟信号数…...
Kotlin delay方法解析
本文记录了kotlin协程(Android)中delay方法的字节码实现,并解析了delay方法如何实现挂起操作。 一、delay方法介绍 1.1、delay方法使用举例 class TestDelay {suspend fun testDelay() {Log.d("TestDelay", "before delay")delay(1000)Log.d…...
PHP框架在大规模分布式系统中的适用性如何?
随着互联网业务的指数级增长,大规模分布式系统已成为支撑高并发、高可用服务的核心技术架构,同时也成为众多互联网企业的首选架构。本文将带大家全面剖析PHP框架在分布式系统中的适用性,并结合实战案例帮大家提供技术选型建议。 一、PHP框架…...
【Vulkan 入门系列】创建描述符集布局和图形管线(五)
描述符集布局定义了着色器如何访问资源(如缓冲区和图像),是渲染管线配置的关键部分。图形管线定义了从顶点数据到最终像素输出的整个处理流程,包括可编程阶段(如顶点和片段着色器)和固定功能阶段࿰…...
Web前端:百度首页克隆 - 前端开发练习
一、项目概述 1.1 练习目标:通过实现百度首页经典布局掌握HTMLCSS基础布局能力 1.2 功能要求: 顶部导航栏布局中央搜索区布局底部信息栏布局基础交互效果 二、技术栈 HTML5 语义化标签CSS3 样式传统布局方案(浮动布局)基础CSS…...
mysql中in的用法详解
MySQL 中 IN 操作符用法详解 IN 是 MySQL 中用于多值筛选的高效操作符,常用于 WHERE 子句,可替代多个 OR 条件,简化查询逻辑并提升可读性。以下从基础语法、应用场景、性能优化、常见问题及高级技巧进行全方位解析。 一、基础语法与优势 1.…...
MySQL为什么默认使用RR隔离级别?
大家好,我是锋哥。今天分享关于【MySQL为什么默认使用RR隔离级别?】面试题。希望对大家有帮助; MySQL为什么默认使用RR隔离级别? 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 MySQL 默认使用 RR(Repeatable Read)…...
施磊老师基于muduo网络库的集群聊天服务器(二)
文章目录 Cmake简单介绍Cmake与MakefileCmake配置CmakeLists.txt 编写完整cmake例子文件夹杂乱问题多级目录Cmakevscode 极其推荐 的 cmake方式 Mysql环境与编程mysql简单使用User表Friend表AllGroup表GroupUser表OfflineMessage表 集群聊天项目工程目录创建网络模块代码Chatse…...
Git拉分支技巧:从零开始创建并推送分支
Git拉分支技巧:从零开始创建并推送分支 在团队协作开发中,Git 分支管理是不可或缺的技能。合理地创建、同步和推送分支,不仅能提高开发效率,还能避免代码冲突。本文将基于以下技巧,详细讲解如何从零开始创建并推送一个…...
Kotlin实现Android应用保活方案
Kotlin实现Android应用保活优化方案 以下的Android应用保活实现方案,更加符合现代Android开发规范,同时平衡系统限制和用户体验。 1. 前台服务方案 class OptimizedForegroundService : Service() {private val notificationId 1private val channel…...
Mysql insert一条数据的详细过程
以下是MySQL在接收到INSERT语句后存储数据的详细过程解析,结合存储引擎(以InnoDB为例)和物理存储机制分步说明。 一、SQL解析与事务启动 1.语法解析 MySQL首先解析INSERT语句,验证字段是否存在、数据类型是否匹配、约束…...
线性DP:最长上升子序列(子序列可不连续,子数组必须连续)
目录 Q1:简单遍历 Q2:变式(加大数据量) Q1:简单遍历 Dp问题 状态表示 f(i,j) 集合所有以第i个数结尾的上升子序列集合-f(i,j)的值存的是什么序列长度最大值max- 状态计算 (其实质是集合的划分)…...
C语言之文本加密程序设计
🌟 嗨,我是LucianaiB! 🌍 总有人间一两风,填我十万八千梦。 🚀 路漫漫其修远兮,吾将上下而求索。 文本加密程序设计 摘要:本文设计了一种文本加密程序,旨在提高信息安…...
生成器模式深入解析与 Spring 源码应用
摘要 本文以生成器模式为研究对象,采用通俗易懂的表述方式,详细阐释其核心概念与运行机制。通过构建游戏角色创建、电商订单生成等实际 Java 案例,直观呈现该模式在复杂对象构建中的应用优势。同时,深入剖析 Spring 框架源码&…...
云效部署实现Java项目自动化部署图解
前言 记录下使用云效部署Java项目,实现java项目一键化自动化部署。 云效流程说明: 1.云效拉取最新git代码后 2.进行maven编译打包后,上传到指定服务器目录 3.通过shell脚本,先kill java项目后,通过java -jar 启动项…...
0801ajax_mock-网络ajax请求1-react-仿低代码平台项目
0 vite配置proxy代理 vite.config.ts代码如下图所示: import { defineConfig } from "vite"; import react from "vitejs/plugin-react";// https://vite.dev/config/ export default defineConfig({plugins: [react()],server: {proxy: {&qu…...
基于Python智能体API的Word自动化排版系统:从零构建全流程模块化工作流与版本控制研究
基于Python智能体API的Word自动化排版系统:从零构建全流程模块化工作流与版本控制实践研究 1. 引言2. 研究背景与意义3. 自动排版工作流的设计原理3.1 文档内容提取与解析3.2 样式参数与格式化规则3.3 智能体API接口调用3.4 自动生成与批量处理3.5 与生成式AI的协同4. 系统架构…...
Java【网络原理】(4)HTTP协议
目录 1.前言 2.正文 2.1自定义协议 2.2HTTP协议 2.2.1抓包工具 2.2.2请求响应格式 2.2.2.1URL 2.2.2.2urlencode 2.2.3认识方法 2.2.3.1GET与POST 2.2.3.2PUT与DELETE 2.2.4请求头关键属性 3.小结 1.前言 哈喽大家好啊,今天来继续给大家带来Java中网络…...
每天学一个 Linux 命令(27):head
可访问网站查看,视觉品味拉满: http://www.616vip.cn/27/index.html head 是 Linux 中用于查看文件开头部分内容的命令,默认显示文件前 10 行,适合快速预览文件结构或日志头部信息。 命令格式 head [选项] [文件]常用选项 选项说明-n <行数>指定显示前 N 行(如…...
【2025软考高级架构师】——计算机系统基础(7)
摘要 本文主要介绍了计算机系统的组成,包括硬件和软件两大部分。硬件由处理器、存储器、总线、接口和外部设备等组成,软件则涵盖系统软件和应用软件。文章还详细阐述了冯诺依曼计算机的组成结构,包括 CPU、主存储器、外存等,并解…...
自定义 strlen 函数:递归实现字符串长度计算
目录 自定义 strlen 函数:递归实现字符串长度计算 一.引言 二.代码呈现 三.代码结构与功能概述 1.自定义 my_strlen 函数 1.函数参数与功能 2.代码逻辑分析 1.参数有效性检查: 2.递归计算字符串长度: 2.main 函数 1.变量定义 2.函…...
LeetCode 打家劫舍+删除并获得点数
题目描述 打家劫舍题目传送门1 删除并获得点数传送门2 思路 这两道题看似毫无关系,但竟然可以用桶数组联系起来!! 先说打家劫舍这道题 限制条件是不能走相邻的屋,再联想到跳台阶(走一格或两格)&#x…...
