寫一個(gè)hashmap

jdk的hashmap悄谐,當(dāng)然比較完美,這里寫一個(gè)簡(jiǎn)單的hashmap库北。

第一版本:

1爬舰、hash算法沒有hashmap好。
2寒瓦、數(shù)組長(zhǎng)度沒有做到2的n次方情屹。
3、沒有jdk8的紅黑樹杂腰。
4垃你、擴(kuò)容算法,直接粗暴的重新放到新數(shù)組喂很,做了一些無用功惜颇,比如,擴(kuò)容放元素的時(shí)候少辣,是不需要比較的凌摄,可以直接放置在鏈表的頭部,不需要一個(gè)一個(gè)遍歷比較漓帅,然后放到尾部锨亏。
4、等等忙干。

public class DiyHashMap<K, V> {

    private DiyHashMap.Node[] nodes;


    private int size;

    private Integer threshold;

    private Float rate;

    private Integer length;


    public DiyHashMap(Integer length, Float rate) {
        this.length = length;
        this.rate = rate;
        Float temp = length * rate;
        this.threshold = temp.intValue();
    }


    private class Node {
        private K key;
        private V value;
        private Node next;

        public Node(K key, V value) {
            this.key = key;
            this.value = value;
        }
    }

    public void put(K k, V v) {
        if (nodes == null) {
            nodes = new DiyHashMap.Node[length];
        }
        if (size >= threshold) {
            // 擴(kuò)容
            resize();
        }
        Node node = new Node(k, v);
        int index = indexOfArray(k.hashCode());
        if (nodes[index] == null) {
            nodes[index] = node;
            size++;
        } else {
            // 鏈表查找
            Node n = nodes[index];
            while (true) {
                if (n.next == null) {
                    // 直接隊(duì)列尾放置
                    n.next = node;
                    size++;
                    break;
                } else if (n.next.key.hashCode() == k.hashCode() && n.next.key.equals(k)) {
                    // 替換
                    n.next.value = v;
                    break;
                } else {
                    // 往下遍歷
                    n = n.next;
                }
            }
        }
    }

    public void resize() {
        int newLength = length << 1;
        this.length = newLength;
        Float temp = length * rate;
        this.threshold = temp.intValue();
        this.size = 0;
        Node[] oldNodes = nodes;
        DiyHashMap.Node[] newNodes = new DiyHashMap.Node[newLength];
        this.nodes = newNodes;

        for (int i = 0; i < oldNodes.length; i++) {
            Node oldNode = oldNodes[i];
            while (true) {
                if (oldNode != null) {
                    this.put(oldNode.key, oldNode.value);
                    oldNode = oldNode.next;
                } else {
                    break;
                }
            }
        }

    }

    public int indexOfArray(int hashcode) {
        int i = hashcode % length;
        if(i < 0){
            return 0 - i;
        }
        return i;
    }

    public V get(K k) {
        int index = indexOfArray(k.hashCode());
        Node node = nodes[index];
        while (true) {
            if (node == null) {
                return null;
            } else if (node.key.hashCode() == k.hashCode() && node.key.equals(k)) {
                // 找到了
                return node.value;
            } else {
                node = node.next;
            }
        }
    }
}

測(cè)試:正確

 public static void main(String[] args) {
        DiyHashMap<String, String> map = new DiyHashMap<>(2, 0.75f);
        String key = "hello";
        for(int i = 0; i < 10000; i++){
            map.put(key + i, "world" + i);

        }

        for(int i = 0; i < 10000; i++){
            String value = map.get(key + i);
            if(!value.equals("world" + i)){
                System.out.println("fail");
            }
        }
    }

再來一個(gè)版本:
1屯伞、優(yōu)化了一下擴(kuò)容。
2豪直、因?yàn)?的n次方劣摇,需要比較麻煩的位運(yùn)算,所以除非復(fù)制hashmap的代碼弓乙,否則末融,手寫不出來。暇韧。
2的n次方寫不出勾习,那么取模運(yùn)算,也就優(yōu)化不了懈玻。
2的n次方寫不出來巧婶,就不能用高位,去推算擴(kuò)容后的位置,所以擴(kuò)容后也就只能鏈表倒置了艺栈。
3英岭、紅黑樹太復(fù)雜了,也不寫湿右。

public class DiyHashMapV2<K, V> {
    private DiyHashMapV2.Node[] nodes;
    private int size;
    private Integer threshold;

    private Float rate;

    private Integer length;


    public DiyHashMapV2(Integer length, Float rate) {
        this.length = length;
        this.rate = rate;
        Float temp = length * rate;
        this.threshold = temp.intValue();
    }


    private class Node {
        private K key;
        private V value;
        private Node next;

        public Node(K key, V value) {
            this.key = key;
            this.value = value;
        }
    }

    public void put(K k, V v) {
        if (nodes == null) {
            nodes = new DiyHashMapV2.Node[length];
        }
        if (size >= threshold) {
            // 擴(kuò)容
            resize();
        }
        Node node = new Node(k, v);
        int index = indexOfArray(k.hashCode());
        if (nodes[index] == null) {
            nodes[index] = node;
            size++;
        } else {
            // 鏈表查找
            Node n = nodes[index];
            while (true) {
                if (n.next == null) {
                    // 直接隊(duì)列尾放置
                    n.next = node;
                    size++;
                    break;
                } else if (n.next.key.hashCode() == k.hashCode() && n.next.key.equals(k)) {
                    // 替換
                    n.next.value = v;
                    break;
                } else {
                    // 往下遍歷
                    n = n.next;
                }
            }
        }
    }
// 就重寫了一下擴(kuò)容方法
    public void resize() {
        int newLength = length << 1;
        this.length = newLength;
        Float temp = length * rate;
        this.threshold = temp.intValue();
        Node[] oldNodes = nodes;
        DiyHashMapV2.Node[] newNodes = new DiyHashMapV2.Node[newLength];
        this.nodes = newNodes;
        for (int i = 0; i < oldNodes.length; i++) {
            Node oldNode = oldNodes[i];
            while (true) {
                if (oldNode != null) {
                    int newIndex = indexOfArray(oldNode.key.hashCode());
                    // 新數(shù)組的這個(gè)位置是否有值
                    if (null == newNodes[newIndex]) {
                        newNodes[newIndex] = oldNode;
                        oldNode = oldNode.next;
                        // 這里需要把以前的next指標(biāo)清空诅妹,否則,因?yàn)橐闳耍瑪U(kuò)容后吭狡,鏈表倒置了,會(huì)造成鏈表循環(huán)next丈莺,也就是鏈表的最后一個(gè)元素的next會(huì)指向鏈表的第一個(gè)元素
                        newNodes[newIndex].next = null;
                    } else {
                        // 直接放到頭部
                        Node tmp = oldNode.next;
                        Node t = newNodes[newIndex];
                        newNodes[newIndex] = oldNode;
                        oldNode.next = t;
                        oldNode = tmp;
                    }
                } else {
                    break;
                }

            }

        }
    }

    public int indexOfArray(int hashcode) {
        int i = hashcode % length;
        if(i < 0){
            return 0 - i;
        }
        return i;
//        return 1;
    }

    public V get(K k) {
        int index = indexOfArray(k.hashCode());
        Node node = nodes[index];
        while (true) {
            if (node == null) {
                return null;
            } else if (node.key.hashCode() == k.hashCode() && node.key.equals(k)) {
                // 找到了
                return node.value;
            } else {
                node = node.next;
            }
        }
    }
}

測(cè)試一下性能:
測(cè)試結(jié)果其實(shí)性能差不多划煮。
測(cè)試的結(jié)果不是很精確,因?yàn)槊看味疾灰粯拥薅恚愦耍琧pu的波動(dòng),分配的執(zhí)行時(shí)間有關(guān)牵现。

public static void main(String[] args) {
        String key = "hello";
        DiyHashMap<String, String> map = new DiyHashMap<>(16, 0.75f);
        for (int i = 0; i < 5000000; i++) {
            map.put(key + i, "world" + i);
        }

        long start = System.nanoTime();
        DiyHashMap<String, String> map1 = new DiyHashMap<>(16, 0.75f);
        for (int i = 0; i < 5000000; i++) {
            map1.put(key + i, "world" + i);

        }
        /*
測(cè)試铐懊,能不能取到元素
for (int i = 0; i < 10000; i++) {
            String value = map.get(key + i);
            if (!value.equals("world" + i)) {
                System.out.println("fail");
            }
        }*/
        long end = System.nanoTime();
        DiyHashMapV2<String, String> map2 = new DiyHashMapV2<>(16, 0.75f);
        for (int i = 0; i < 5000000; i++) {
            map2.put(key + i, "world" + i);
        }
        long end2 = System.nanoTime();

        Map<String, String> map3 = new HashMap<>(16, 0.75f);
        for (int i = 0; i < 5000000; i++) {
            map3.put(key + i, "world" + i);
        }
        long end3 = System.nanoTime();
        System.out.println(end - start);
        System.out.println(end2 - end);
        System.out.println(end3 - end2);

   }

3128620097
5787549157
3656479197
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市瞎疼,隨后出現(xiàn)的幾起案子科乎,更是在濱河造成了極大的恐慌,老刑警劉巖贼急,帶你破解...
    沈念sama閱讀 217,509評(píng)論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件茅茂,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡太抓,警方通過查閱死者的電腦和手機(jī)空闲,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,806評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來走敌,“玉大人碴倾,你說我怎么就攤上這事〉衾觯” “怎么了跌榔?”我有些...
    開封第一講書人閱讀 163,875評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)捶障。 經(jīng)常有香客問我僧须,道長(zhǎng),這世上最難降的妖魔是什么项炼? 我笑而不...
    開封第一講書人閱讀 58,441評(píng)論 1 293
  • 正文 為了忘掉前任担平,我火速辦了婚禮示绊,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘暂论。我一直安慰自己面褐,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,488評(píng)論 6 392
  • 文/花漫 我一把揭開白布空另。 她就那樣靜靜地躺著盆耽,像睡著了一般蹋砚。 火紅的嫁衣襯著肌膚如雪扼菠。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,365評(píng)論 1 302
  • 那天坝咐,我揣著相機(jī)與錄音循榆,去河邊找鬼。 笑死墨坚,一個(gè)胖子當(dāng)著我的面吹牛秧饮,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播泽篮,決...
    沈念sama閱讀 40,190評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼盗尸,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了帽撑?” 一聲冷哼從身側(cè)響起泼各,我...
    開封第一講書人閱讀 39,062評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎亏拉,沒想到半個(gè)月后扣蜻,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,500評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡及塘,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,706評(píng)論 3 335
  • 正文 我和宋清朗相戀三年莽使,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片笙僚。...
    茶點(diǎn)故事閱讀 39,834評(píng)論 1 347
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡芳肌,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出肋层,到底是詐尸還是另有隱情庇勃,我是刑警寧澤,帶...
    沈念sama閱讀 35,559評(píng)論 5 345
  • 正文 年R本政府宣布槽驶,位于F島的核電站责嚷,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏掂铐。R本人自食惡果不足惜罕拂,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,167評(píng)論 3 328
  • 文/蒙蒙 一揍异、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧爆班,春花似錦衷掷、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,779評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至枢舶,卻和暖如春懦胞,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背凉泄。 一陣腳步聲響...
    開封第一講書人閱讀 32,912評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工躏尉, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人后众。 一個(gè)月前我還...
    沈念sama閱讀 47,958評(píng)論 2 370
  • 正文 我出身青樓胀糜,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親蒂誉。 傳聞我的和親對(duì)象是個(gè)殘疾皇子教藻,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,779評(píng)論 2 354

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