ConsistentHash一致性Hash——Java實(shí)現(xiàn)


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);

    }

}


?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末拘悦,一起剝皮案震驚了整個(gè)濱河市凯正,隨后出現(xiàn)的幾起案子葱绒,更是在濱河造成了極大的恐慌凄诞,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,539評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件由桌,死亡現(xiàn)場離奇詭異为黎,居然都是意外死亡邮丰,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,594評(píng)論 3 396
  • 文/潘曉璐 我一進(jìn)店門铭乾,熙熙樓的掌柜王于貴愁眉苦臉地迎上來剪廉,“玉大人,你說我怎么就攤上這事炕檩《方” “怎么了?”我有些...
    開封第一講書人閱讀 165,871評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵笛质,是天一觀的道長泉沾。 經(jīng)常有香客問我,道長妇押,這世上最難降的妖魔是什么跷究? 我笑而不...
    開封第一講書人閱讀 58,963評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮敲霍,結(jié)果婚禮上揭朝,老公的妹妹穿的比我還像新娘。我一直安慰自己色冀,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,984評(píng)論 6 393
  • 文/花漫 我一把揭開白布柱嫌。 她就那樣靜靜地躺著锋恬,像睡著了一般。 火紅的嫁衣襯著肌膚如雪编丘。 梳的紋絲不亂的頭發(fā)上与学,一...
    開封第一講書人閱讀 51,763評(píng)論 1 307
  • 那天,我揣著相機(jī)與錄音嘉抓,去河邊找鬼索守。 笑死,一個(gè)胖子當(dāng)著我的面吹牛抑片,可吹牛的內(nèi)容都是我干的卵佛。 我是一名探鬼主播,決...
    沈念sama閱讀 40,468評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼敞斋,長吁一口氣:“原來是場噩夢啊……” “哼截汪!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起植捎,我...
    開封第一講書人閱讀 39,357評(píng)論 0 276
  • 序言:老撾萬榮一對情侶失蹤衙解,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后焰枢,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蚓峦,經(jīng)...
    沈念sama閱讀 45,850評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡舌剂,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,002評(píng)論 3 338
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了暑椰。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片霍转。...
    茶點(diǎn)故事閱讀 40,144評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖干茉,靈堂內(nèi)的尸體忽然破棺而出谴忧,到底是詐尸還是另有隱情,我是刑警寧澤角虫,帶...
    沈念sama閱讀 35,823評(píng)論 5 346
  • 正文 年R本政府宣布沾谓,位于F島的核電站,受9級(jí)特大地震影響戳鹅,放射性物質(zhì)發(fā)生泄漏均驶。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,483評(píng)論 3 331
  • 文/蒙蒙 一枫虏、第九天 我趴在偏房一處隱蔽的房頂上張望妇穴。 院中可真熱鬧,春花似錦隶债、人聲如沸腾它。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,026評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽瞒滴。三九已至,卻和暖如春赞警,著一層夾襖步出監(jiān)牢的瞬間妓忍,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,150評(píng)論 1 272
  • 我被黑心中介騙來泰國打工愧旦, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留世剖,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,415評(píng)論 3 373
  • 正文 我出身青樓笤虫,卻偏偏與公主長得像旁瘫,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子琼蚯,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,092評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容