成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

HashMap源碼閱讀小記

blastz / 1805人閱讀

摘要:否則,繼續(xù)判斷頭節(jié)點(diǎn)是否是的實(shí)例,是一個(gè)紅黑樹,若是,則直接在樹中插入。在中有一個(gè)屬性為,這是一個(gè)閾值,若數(shù)量超過它,鏈表會(huì)轉(zhuǎn)化為紅黑樹,小于它則會(huì)換回鏈表。所以同時(shí)用到了數(shù)組,鏈表,紅黑樹這三種數(shù)據(jù)結(jié)構(gòu)。

1. HashMap中Node類:
static class Node implements Map.Entry {
        final int hash;
        final K key;
        V value;
        Node next;

        Node(int hash, K key, V value, Node next) {
            this.hash = hash;
            this.key = key;
            this.value = value;
            this.next = next;
        }

        public final K getKey()        { return key; }
        public final V getValue()      { return value; }
        public final String toString() { return key + "=" + value; }

        public final int hashCode() {
            return Objects.hashCode(key) ^ Objects.hashCode(value);
        }

        public final V setValue(V newValue) {
            V oldValue = value;
            value = newValue;
            return oldValue;
        }

        public final boolean equals(Object o) {
            if (o == this)
                return true;
            if (o instanceof Map.Entry) {
                Map.Entry e = (Map.Entry)o;
                if (Objects.equals(key, e.getKey()) &&
                    Objects.equals(value, e.getValue()))
                    return true;
            }
            return false;
        }
    }

重寫hashCode,key和value的hashcode取異或。

重寫equals,當(dāng)為同一個(gè)對象或是同一個(gè)key和同一個(gè)value都認(rèn)為這兩個(gè)對象相等。

2.散列值的計(jì)算
static final int hash(Object key) {
        int h;
        return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
    }

與無符號右移的自己異或同時(shí)兼顧了hash高16位和低16位,讓散列值更散。

3. 關(guān)注 get(Object key)
public V get(Object key) {
        Node e;
        return (e = getNode(hash(key), key)) == null ? null : e.value;
    }
    
final Node getNode(int hash, Object key) {
        Node[] tab; Node first, e; int n; K k;
        if ((tab = table) != null && (n = tab.length) > 0 &&
            (first = tab[(n - 1) & hash]) != null) {
            if (first.hash == hash && // always check first node
                ((k = first.key) == key || (key != null && key.equals(k))))
                return first;
            if ((e = first.next) != null) {
                if (first instanceof TreeNode)
                    return ((TreeNode)first).getTreeNode(hash, key);
                do {
                    if (e.hash == hash &&
                        ((k = e.key) == key || (key != null && key.equals(k))))
                        return e;
                } while ((e = e.next) != null);
            }
        }
        return null;
    }

可以看出,get()是拿著key的hash和key去找的值。

在getNode()中先是一系列判斷和賦值,然后通過下標(biāo)找定位到key在table中的位置

定位的方式:(n - 1) & hash,這樣取出的值總是小于table長度n的。

然后對比key是否相等,相等就返回,不相等則判斷是否是紅黑樹存儲(chǔ)結(jié)構(gòu),若是則在紅黑樹上查找。

若不是則在鏈表結(jié)構(gòu)上查找。

4.核心put(K key, V value)
public V put(K key, V value) {
        return putVal(hash(key), key, value, false, true);
    }
    
    final V putVal(int hash, K key, V value, boolean onlyIfAbsent,
                   boolean evict) {
        Node[] tab; Node p; int n, i;
        if ((tab = table) == null || (n = tab.length) == 0)
            n = (tab = resize()).length;
        if ((p = tab[i = (n - 1) & hash]) == null)
            tab[i] = newNode(hash, key, value, null);
        else {
            Node e; K k;
            if (p.hash == hash &&
                ((k = p.key) == key || (key != null && key.equals(k))))
                e = p;
            else if (p instanceof TreeNode)
                e = ((TreeNode)p).putTreeVal(this, tab, hash, key, value);
            else {
                for (int binCount = 0; ; ++binCount) {
                    if ((e = p.next) == null) {
                        p.next = newNode(hash, key, value, null);
                        if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st
                            treeifyBin(tab, hash);
                        break;
                    }
                    if (e.hash == hash &&
                        ((k = e.key) == key || (key != null && key.equals(k))))
                        break;
                    p = e;
                }
            }
            if (e != null) { // existing mapping for key
                V oldValue = e.value;
                if (!onlyIfAbsent || oldValue == null)
                    e.value = value;
                afterNodeAccess(e);
                return oldValue;
            }
        }
        ++modCount;
        //判斷是否擴(kuò)容
        if (++size > threshold)
            resize();
        afterNodeInsertion(evict);
        return null;
    }

首先調(diào)用了putVal(int hash, K key, V value, boolean onlyIfAbsent,boolean evict),

第一步做初始化工作。

然后,定位到table沒有沖突,則直接存到table上。

若是沖突了,則判斷key是否相等,若相等則,直接將舊德的Node覆蓋。

否則,繼續(xù)判斷頭節(jié)點(diǎn)是否是TreeNode的實(shí)例,TreeNode是一個(gè)紅黑樹,若是,則直接在樹中插入。

如果不是紅黑樹,插到鏈表的尾部。

在hashmap中有一個(gè)屬性為TREEIFY_THRESHOLD,這是一個(gè)閾值,若數(shù)量超過它,鏈表會(huì)轉(zhuǎn)化為紅黑樹,小于它則會(huì)換回鏈表。所以hashMap同時(shí)用到了數(shù)組,鏈表,紅黑樹這三種數(shù)據(jù)結(jié)構(gòu)。

每次新添一個(gè)節(jié)點(diǎn)都會(huì)判斷是否需要擴(kuò)容。

5. 擴(kuò)容機(jī)制resize()

首先涉及三個(gè)成員變量:

capacity:容量

loadFactor:裝載因子(0-1之間)

threshold:判斷是否需要擴(kuò)容的標(biāo)志threshold = capacity * loadFactor

所以裝載因子控制著HashMap沖突比例。

每次擴(kuò)容都擴(kuò)大到之前的兩倍。

擴(kuò)容會(huì)重新建table等變量,因此開銷比較大。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/74231.html

相關(guān)文章

  • 集合小記

    摘要:解決沖突開放定址法拉鏈法表解決沖突開放定址法再哈希法鏈地址法建立公共溢出區(qū)并發(fā)包中的線程安全的集合容器線程安全的,不允許為,默認(rèn)個(gè)的數(shù)組,每個(gè)中實(shí)現(xiàn)就是了,通過定位。基于數(shù)組,線程安全的集合類,容量可以限制。 List   List?元素是有序的、可重復(fù),實(shí)現(xiàn)List接口的集合主要有:ArrayList、LinkedList、Vector、Stack。   ArrayList:動(dòng)態(tài)數(shù)組...

    alaege 評論0 收藏0
  • HashSet源碼分析:JDK源碼系列

    摘要:簡介繼續(xù)分析源碼,上一篇文章把的分析完畢。本文開始分析簡單的介紹一下。存儲(chǔ)的元素是無序的并且允許使用空的元素。 1.簡介 繼續(xù)分析源碼,上一篇文章把HashMap的分析完畢。本文開始分析HashSet簡單的介紹一下。 HashSet是一個(gè)無重復(fù)元素集合,內(nèi)部使用HashMap實(shí)現(xiàn),所以HashMap的特征耶繼承了下來。存儲(chǔ)的元素是無序的并且HashSet允許使用空的元素。 HashSe...

    用戶83 評論0 收藏0
  • Node程序debug小記

    摘要:當(dāng)前的部分代碼狀態(tài)超時(shí)再縮小了范圍以后,進(jìn)一步進(jìn)行排查。函數(shù)是一個(gè)很簡單的一次性函數(shù),在第一次被觸發(fā)時(shí)調(diào)用函數(shù)。因?yàn)樯鲜鍪褂玫氖?,而非,所以在獲取的時(shí)候,肯定為空,那么這就意味著會(huì)繼續(xù)調(diào)用函數(shù)。 有時(shí)候,所見并不是所得,有些包,你需要去翻他的源碼才知道為什么會(huì)這樣。 背景 今天調(diào)試一個(gè)程序,用到了一個(gè)很久之前的NPM包,名為formstream,用來將form表單數(shù)據(jù)轉(zhuǎn)換為流的形式進(jìn)行...

    Achilles 評論0 收藏0
  • 淺析HashMap源碼(1)

    摘要:前言本文的目的是閱讀理解的源碼,作為集合中重要的一個(gè)角色,平時(shí)用到十分多的一個(gè)類,深入理解它,知其所以然很重要。 前言 本文的目的是閱讀理解HashMap的源碼,作為集合中重要的一個(gè)角色,平時(shí)用到十分多的一個(gè)類,深入理解它,知其所以然很重要。本文基于Jdk1.7,因?yàn)镴dk1.8改變了HashMap的數(shù)據(jù)結(jié)構(gòu),進(jìn)行了優(yōu)化,我們先從基礎(chǔ)閱讀,之后再閱讀理解Jdk1.8的內(nèi)容 HashMa...

    wwolf 評論0 收藏0
  • JDK1.8 ArrayList部分源碼分析小記

    摘要:部分源碼分析小記底層數(shù)據(jù)結(jié)構(gòu)底層的數(shù)據(jù)結(jié)構(gòu)就是數(shù)組,數(shù)組元素類型為類型,即可以存放所有類型數(shù)據(jù)。初始容量大于初始化元素?cái)?shù)組新建一個(gè)數(shù)組初始容量為為空對象數(shù)組初始容量小于,拋出異常無參構(gòu)造函數(shù)。 JDK1.8 ArrayList部分源碼分析小記 底層數(shù)據(jù)結(jié)構(gòu) 底層的數(shù)據(jù)結(jié)構(gòu)就是數(shù)組,數(shù)組元素類型為Object類型,即可以存放所有類型數(shù)據(jù)。我們對ArrayList類的實(shí)例的所有的操作底層都...

    王軍 評論0 收藏0

發(fā)表評論

0條評論

最新活動(dòng)
閱讀需要支付1元查看
<