All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.github.icodegarden.nutrient.lang.algorithm.consistenthash.ConsistentHashRouter Maven / Gradle / Ivy

There is a newer version: 3.0.2
Show newest version
package io.github.icodegarden.nutrient.lang.algorithm.consistenthash;

import java.util.Collection;
import java.util.Iterator;
import java.util.Map.Entry;

import io.github.icodegarden.nutrient.lang.algorithm.HashFunction;
import io.github.icodegarden.nutrient.lang.algorithm.MD5Function;

import java.util.SortedMap;
import java.util.TreeMap;

/**
 * 将节点对象散列到具有一定数量虚拟节点的散列环中
* 一致性hash的目的是当物理节点删除时,只需调整落到该节点的数据到新的物理节点,避免大规模的调整引起系统的不稳定 * * @author Fangfang.Xu */ public class ConsistentHashRouter { private final SortedMap> ring = new TreeMap>(); private final HashFunction hashFunc; public ConsistentHashRouter(Collection pNodes) { this(pNodes, 160/*默认每个物理节点有160个虚拟节点*/); } public ConsistentHashRouter(Collection pNodes, int vNodeCount) { this(pNodes, vNodeCount, new MD5Function()); } public ConsistentHashRouter(Collection pNodes, int vNodeCount, HashFunction hashFunc) { this.hashFunc = hashFunc; if (pNodes != null) { for (T pNode : pNodes) { addOrUpdatePhysicalNode(pNode, vNodeCount); } } } /** * 在hash环上新增物理节点或在原物理节点上追加vNodeCount数量的虚拟节点
* * 新增:每个虚拟机节点都新建,number递增
* 追加:先查出已存在的虚拟节点数,在这个基础上追加vNodeCount数量的虚拟节点
*/ public void addOrUpdatePhysicalNode(T pNode, int vNodeCount) { if (vNodeCount < 0) throw new IllegalArgumentException("illegal virtual node counts :" + vNodeCount); int existingCount = countExistingVirtualNodes(pNode); for (int i = 0; i < vNodeCount; i++) { VirtualNode vNode = new VirtualNode(pNode, i + existingCount); ring.put(hashFunc.hash(vNode.getKey()), vNode); } } /** * 移除物理节点,把该物理节点对应的虚拟节点从环上remove * * @return 移除的虚拟节点 */ public void removePhysicalNode(T pNode) { Iterator>> it = ring.entrySet().iterator(); while (it.hasNext()) { Entry> next = it.next(); VirtualNode virtualNode = next.getValue(); if (virtualNode.isVirtualNodeOf(pNode)) { it.remove(); } } } /** * 根据key找到该key应该落到的虚拟节点 */ public VirtualNode routeNode(String key) { if (ring.isEmpty()) { return null; } /** * 用跟虚拟节点一样的hash算法 */ Number hashVal = hashFunc.hash(key); /** * tail后,得到的是 >= 传参的hash值的 那段hash环,有可能是空的 */ SortedMap> tailMap = ring.tailMap(hashVal); /** * 不为空,应该落到最近的虚拟节点对应的物理节点; 为空,即相当于要落到整个hash环的第一个虚拟节点对应的物理节点 */ Number nodeHashVal = !tailMap.isEmpty() ? tailMap.firstKey() : ring.firstKey(); return ring.get(nodeHashVal); } /** * 获取物理节点已存在的虚拟节点数量 */ public int countExistingVirtualNodes(T pNode) { int count = 0; for (VirtualNode vNode : ring.values()) { if (vNode.isVirtualNodeOf(pNode)) { count++; } } return count; } /** * 获取所有的虚拟节点数量 */ public int countExistingVirtualNodes() { return ring.size(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy