import java.util.Arrays;
import java.util.Collection;
import java.util.SortedMap;
import java.util.TreeMap;
/**
* Created by 01369537 on 2019/8/16.
*/
public class ConsistentHash<T> {
private final HashFunction hashFunction;
private final int numberOfReplicas;// 節(jié)點(diǎn)的復(fù)制因子,實(shí)際節(jié)點(diǎn)個(gè)數(shù) * numberOfReplicas = 虛擬節(jié)點(diǎn)個(gè)數(shù)
private final SortedMap<Long, T> circle = new TreeMap<>();// 存儲(chǔ)虛擬節(jié)點(diǎn)的hash值到真實(shí)節(jié)點(diǎn)的映射
public ConsistentHash(HashFunction hashFunction, int numberOfReplicas,
Collection<T> nodes) {
this.hashFunction = hashFunction;
this.numberOfReplicas = numberOfReplicas;
for (T node : nodes)
add(node);
}
public void add(T node) {
for (int i = 0; i < numberOfReplicas; i++)
// 對于一個(gè)實(shí)際機(jī)器節(jié)點(diǎn) node, 對應(yīng) numberOfReplicas 個(gè)虛擬節(jié)點(diǎn)
/*
* 不同的虛擬節(jié)點(diǎn)(i不同)有不同的hash值,但都對應(yīng)同一個(gè)實(shí)際機(jī)器node
* 虛擬node一般是均衡分布在環(huán)上的,數(shù)據(jù)存儲(chǔ)在順時(shí)針方向的虛擬node上
*/ {
String key = node.toString() + i;
Long hash = hashFunction.hash(key);
System.out.println("key=" + key + " hash=" + hash);
circle.put(hash, node);
}
}
public void remove(T node) {
for (int i = 0; i < numberOfReplicas; i++)
circle.remove(hashFunction.hash(node.toString() + i));
}
/*
* 獲得一個(gè)最近的順時(shí)針節(jié)點(diǎn),根據(jù)給定的key 取Hash
* 然后再取得順時(shí)針方向上最近的一個(gè)虛擬節(jié)點(diǎn)對應(yīng)的實(shí)際節(jié)點(diǎn)
*/
public T get(Object key) {
if (circle.isEmpty())
return null;
long hash = hashFunction.hash((String) key);// node 用String來表示,獲得node在哈希環(huán)中的hashCode
if (!circle.containsKey(hash)) {//數(shù)據(jù)映射在兩臺(tái)虛擬機(jī)器所在環(huán)之間,就需要按順時(shí)針方向?qū)ふ覚C(jī)器
//SortedMap的tailMap函數(shù)返回Map中所有大于該key的元素組成的子map
SortedMap<Long, T> tailMap = circle.tailMap(hash);
hash = tailMap.isEmpty() ? circle.firstKey() : tailMap.firstKey();
}
//返回存儲(chǔ)key的真實(shí)的節(jié)點(diǎn)
return circle.get(hash);
}
public long getSize() {
return circle.size();
}
public static void main(String[] args) {
ConsistentHash<String> consistentHash = new ConsistentHash<>(new HashFunction() {
@Override
public Long hash(Object key) {
return (long) key.hashCode();
}
}, 2, Arrays.asList("1", "2", "3"));
String key = "13";
System.out.println("keyHashCode=" + key.hashCode());
String node = consistentHash.get(key);
System.out.println("node=" + node);
}
}
ConsistentHash一致性Hash——Java實(shí)現(xiàn)
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
- 文/潘曉璐 我一進(jìn)店門铭乾,熙熙樓的掌柜王于貴愁眉苦臉地迎上來剪廉,“玉大人,你說我怎么就攤上這事炕檩《方” “怎么了?”我有些...
- 文/不壞的土叔 我叫張陵笛质,是天一觀的道長泉沾。 經(jīng)常有香客問我,道長妇押,這世上最難降的妖魔是什么跷究? 我笑而不...
- 正文 為了忘掉前任,我火速辦了婚禮敲霍,結(jié)果婚禮上揭朝,老公的妹妹穿的比我還像新娘。我一直安慰自己色冀,他們只是感情好,可當(dāng)我...
- 文/花漫 我一把揭開白布柱嫌。 她就那樣靜靜地躺著锋恬,像睡著了一般。 火紅的嫁衣襯著肌膚如雪编丘。 梳的紋絲不亂的頭發(fā)上与学,一...
- 文/蒼蘭香墨 我猛地睜開眼敞斋,長吁一口氣:“原來是場噩夢啊……” “哼截汪!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起植捎,我...
- 序言:老撾萬榮一對情侶失蹤衙解,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后焰枢,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蚓峦,經(jīng)...
- 正文 獨(dú)居荒郊野嶺守林人離奇死亡舌剂,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
- 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了暑椰。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片霍转。...
- 正文 年R本政府宣布沾谓,位于F島的核電站,受9級(jí)特大地震影響戳鹅,放射性物質(zhì)發(fā)生泄漏均驶。R本人自食惡果不足惜,卻給世界環(huán)境...
- 文/蒙蒙 一枫虏、第九天 我趴在偏房一處隱蔽的房頂上張望妇穴。 院中可真熱鬧,春花似錦隶债、人聲如沸腾它。這莊子的主人今日做“春日...
- 文/蒼蘭香墨 我抬頭看了看天上的太陽瞒滴。三九已至,卻和暖如春赞警,著一層夾襖步出監(jiān)牢的瞬間妓忍,已是汗流浹背。 一陣腳步聲響...
- 正文 我出身青樓笤虫,卻偏偏與公主長得像旁瘫,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子琼蚯,可洞房花燭夜當(dāng)晚...
推薦閱讀更多精彩內(nèi)容
- 背景 隨著業(yè)務(wù)系統(tǒng)越來越大,我們需要對API的訪問進(jìn)行更多的緩存,使用Redis是一個(gè)很好的解決方案.但是單臺(tái)Re...
- import java.io.UnsupportedEncodingException;import java.s...
- 目錄 ?目錄 ?背景 ?分配方法 ?一致性hash原理 ?使用虛擬節(jié)點(diǎn)解決hash不均勻的問題 ?總結(jié) ?Java...
- 105余杏周檢視 【第4周檢視】 (20171106周一~20171112周日) 一罚拟、【好習(xí)慣踐行】 1. 早睡早...