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

資訊專欄INFORMATION COLUMN

JDK源碼那些事兒之我眼中的HashMap

voyagelab / 2717人閱讀

摘要:接下來就來說下我眼中的。鏈表轉(zhuǎn)換為樹的閾值,超過這個長度的鏈表會被轉(zhuǎn)換為紅黑樹,當然,不止這一個條件,在下面的源碼部分會看到。

源碼部分從HashMap說起是因為筆者看了很多遍這個類的源碼部分,同時感覺網(wǎng)上很多都是粗略的介紹,有些可能還不正確,最后只能自己看源碼來驗證理解,寫下這篇文章一方面是為了促使自己能深入,另一方面也是給一些新人一些指導(dǎo),不求有功,但求無過。有錯誤的地方請在評論中指出,我會及時驗證修改,謝謝。

接下來就來說下我眼中的HashMap。

jdk版本:1.8

在深入源碼之前,了解HashMap的整體結(jié)構(gòu)是非常重要的事情,結(jié)構(gòu)也體現(xiàn)出了源碼中一些對HashMap的操作,結(jié)構(gòu)大致如下:

從上邊的結(jié)構(gòu)圖大家應(yīng)該也能看出來HashMap的實現(xiàn)結(jié)構(gòu):數(shù)組+鏈表+紅黑樹。

看下類注釋,直接看源碼部分最好,可能大多數(shù)都看不明白,這里可以看下別人的翻譯:類注釋翻譯。本文中筆者不打算對紅黑樹部分進行講解說明,插入和刪除操作會引發(fā)各種狀態(tài),需要做對應(yīng)的調(diào)整,之后會多帶帶寫一篇紅黑樹基礎(chǔ),結(jié)合TreeNode來做講解。

先總結(jié)一些名詞概念方便初學(xué)者理解:

1.桶(bucket):數(shù)組中存儲元素的位置,參考結(jié)構(gòu)圖,實際上是數(shù)組中的某個索引下的元素,這個元素有可能是樹的根節(jié)點或者鏈表的首節(jié)點,當然,理解上還是一個鏈表或紅黑樹整體當成桶

2.bin:桶中的每個元素,即紅黑樹中的某個元素或者是鏈表中的某個元素。

除了上邊的名詞,最好還能去理解下哈希表,可以參考下。HashMap也是對哈希表的一種實現(xiàn),簡單理解,可以類比數(shù)學(xué)中的求余操作,對范圍進行固定,將大量的數(shù)據(jù)放入一個有界的范圍中,求余放置,這種操作算是哈希表的一種實現(xiàn)方式。

下面進行源碼部分的說明:

類定義

public class HashMap extends AbstractMap
    implements Map, Cloneable, Serializable

繼承AbstractMap
實現(xiàn)Cloneable接口,提供克隆功能
實現(xiàn)Serializable接口,支持序列化,方便序列化傳輸

這里有個有意思的問題:為什么HashMap繼承了AbstractMap還要實現(xiàn)Map接口?有興趣的可以去看下stackoverflow上的回答:

https://stackoverflow.com/que...

變量說明
    /**
     * Node數(shù)組的默認長度,16
     */
    static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; // aka 16

    /**
     * Node數(shù)組的最大長度,最大擴容長度
     */
 
    /**
     * 默認負載因子
     * 這個是干嘛的呢?
     * 負載因子是哈希表在自動擴容之前能承受容量的一種尺度。
     * 當哈希表的數(shù)目超出了負載因子與當前容量的乘積時,則要對該哈希表進行rehash操作(擴容操作)。
     */
    static final float DEFAULT_LOAD_FACTOR = 0.75f;

    /**
     * 鏈表轉(zhuǎn)換為樹的閾值,超過這個長度的鏈表會被轉(zhuǎn)換為紅黑樹,
     * 當然,不止這一個條件,在下面的源碼部分會看到。
     */
    static final int TREEIFY_THRESHOLD = 8;

    /**
     * 當進行resize操作時,小于這個長度的樹會被轉(zhuǎn)換為鏈表       
     */
    static final int UNTREEIFY_THRESHOLD = 6;

    /**
     * 鏈表被轉(zhuǎn)換成樹形的最小容量,
     * 如果沒有達到這個容量只會執(zhí)行resize進行擴容
     * 可以理解成一種計算規(guī)則
     */
    static final int MIN_TREEIFY_CAPACITY = 64;
    /**
     * 
     * 第一次使用的時候進行初始化,put操作才會初始化對象
     * 調(diào)用構(gòu)造函數(shù)時不會初始化,后面源碼可參考
     */
    transient Node[] table;

    /**
     * 
     * entrySet保存key和value 用于迭代
     */
    transient Set> entrySet;

    /**
     * 
     * 存放元素的個數(shù),但不等于數(shù)組的長度
     */
    transient int size;

    /**
     * 
     * 計數(shù)器,fail-fast機制相關(guān),不詳細介紹,有興趣的自己google下
     * 你可以當成一個在高并發(fā)讀寫操作時的判斷,舉個例子:
     * 一個線程A迭代遍歷a,modCount=expectedModCount值為1,執(zhí)行過程中,一個線程B修改了a,modCount=2,線程A在遍歷時判斷了modCount<>expectedModCount,拋錯
     * 當然,這個只是簡單的檢查,并不能得到保證
     */
    transient int modCount;

    /**
     * 
     * 閾值,當實際大小超過閾值(容量*負載因子)的時候,會進行擴容
     */
    int threshold;

    /**
     * 
     * 負載因子
     */
    final float loadFactor;

在看方法之前先看下Node實現(xiàn):

    /**
     * Node的實現(xiàn) 
     * 看出來是最多實現(xiàn)單向鏈表 僅有一個next引用
     * 比較簡單明了,應(yīng)該都能看明白
     */
    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;
        }
        
        /**
         * Map.Entry 判斷類型
         * 鍵值對進行比較 判斷是否相等
         */
        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;
        }
    }
重點說明

在注釋中我會添加一些標記幫助理清流程,同時方便我后邊總結(jié)對照和參考(例如A1,A2是同一級)。

    /**
     * 負載因子設(shè)置成默認值 0.75f
     * A1
     */
    public HashMap() {
        this.loadFactor = DEFAULT_LOAD_FACTOR; // all other fields defaulted
    }
    /**
     * 初始數(shù)組長度設(shè)置,負載因子默認值
     * A2
     */
    public HashMap(int initialCapacity) {
        this(initialCapacity, DEFAULT_LOAD_FACTOR);
    }
    /**
     * 初始長度和負載因子設(shè)置
     * A2
     */   
    public HashMap(int initialCapacity, float loadFactor) {
        if (initialCapacity < 0)
            throw new IllegalArgumentException("Illegal initial capacity: " +
                                               initialCapacity);
        if (initialCapacity > MAXIMUM_CAPACITY)
            initialCapacity = MAXIMUM_CAPACITY;
        if (loadFactor <= 0 || Float.isNaN(loadFactor))
            throw new IllegalArgumentException("Illegal load factor: " +
                                               loadFactor);
        this.loadFactor = loadFactor;
        // 根據(jù)初始容量設(shè)置閾值
        // 二進制操作,比較繞,需要自己好好理解下
        // 這值在resize有用,resize代碼可以注意下,主要是為了區(qū)分是否是有參構(gòu)造函數(shù)還是無參構(gòu)造函數(shù)以便之后的操作
        // 可以參考文章:https://www.cnblogs.com/liujinhong/p/6576543.html
        // 是否有更深層次的考慮筆者還未想到,有大神可以在評論區(qū)告知我
        this.threshold = tableSizeFor(initialCapacity);
    }
    /**
     * 將m存入當前map中
     */  
    public HashMap(Map m) {
        this.loadFactor = DEFAULT_LOAD_FACTOR;
        putMapEntries(m, false);
    }
    /**
     * evict參數(shù)相當于占位符,是為了擴展性,可以追溯到afterNodeInsertion(evict),方法是空的
     * 在LinkedHashMap中有實現(xiàn),有興趣可以去看看
     */  
    final void putMapEntries(Map m, boolean evict) {
        int s = m.size();
        if (s > 0) {
            /**
             * 判斷table是否已經(jīng)被初始化
             */
            if (table == null) { // pre-size
                // 未被初始化,判斷m中元素的個數(shù)放入當前map中是否會超出最大容量的閾值
                float ft = ((float)s / loadFactor) + 1.0F;
                int t = ((ft < (float)MAXIMUM_CAPACITY) ?
                         (int)ft : MAXIMUM_CAPACITY);
                // 計算得到的t大于閾值 閾值設(shè)置
                if (t > threshold)
                    threshold = tableSizeFor(t);
            }
            else if (s > threshold)
                /**
                 * 當前map已經(jīng)初始化,并且添加的元素長度大于閾值,需要進行擴容操作
                 */  
                resize();
            /**
             * 上邊已經(jīng)初始化并處理好閾值設(shè)置,下面使用entrySet循環(huán)putVal保存m中的Node對象的key和value
             * 這里有個重要的地方,
             * putVal的第一個參數(shù),hash(key),map的put操作也是同樣的調(diào)用方式
             * 可以參考文章:https://www.cnblogs.com/liujinhong/p/6576543.html
             * 順便看下源碼上的注釋,主要是減少沖突和性能上的考慮
             */  
            for (Map.Entry e : m.entrySet()) {
                K key = e.getKey();
                V value = e.getValue();
                putVal(hash(key), key, value, false, evict);
            }
        }
    }
    
    /**
     * 擴容操作,重點部分
     * 
     * 如果第一次帶容量參數(shù)時,創(chuàng)建時閾值設(shè)置為對應(yīng)容量的最小的2的N次方(大于等于傳入容量參數(shù)),去看下上邊HashMap(int initialCapacity),
     * 如果添加一個元素,會執(zhí)行resize將閾值設(shè)置為了閾值 * 負載因子,
     * 比如設(shè)置1000 創(chuàng)建時閾值threshold=1024,負載因子默認,其他值都未進行操作,
     * 添加一個元素 閾值變?yōu)?024 * 0.75 = 768,創(chuàng)建的Node數(shù)組長度為1024,size=1,
     * 添加第769個元素時,進行resize操作,threshold=1536,Node數(shù)組長度為2048,數(shù)組拷貝到新數(shù)組中,
     * 如果有確認的數(shù)據(jù)長度,不想讓HashMap進行擴容操作,那么則需要在構(gòu)造時填上計算好的數(shù)組容量
     * 強烈建議自己寫代碼debug試試
     */
    final Node[] resize() {
        //oldTab 保存擴容前的Node數(shù)組
        Node[] oldTab = table;
        // oldCap null的話即為0,否則就是擴容前的Node數(shù)組的容量大小
        int oldCap = (oldTab == null) ? 0 : oldTab.length;
        // 擴容前的閾值
        int oldThr = threshold;
        // 擴容后的數(shù)組容量(長度),擴容后的閾值
        int newCap, newThr = 0;
        // 1.擴容前的數(shù)組不為空
        // B1
        if (oldCap > 0) {
            // 擴容前的Node數(shù)組容量大于等于設(shè)置的最大容量,不會進行擴容,閾值設(shè)置為Integer.MAX_VALUE
            if (oldCap >= MAXIMUM_CAPACITY) {
                // C1
                threshold = Integer.MAX_VALUE;
                return oldTab;
            }
            // 如果擴容前的數(shù)組容量擴大為2倍依然沒有超過最大容量,
            // 并且擴容前的Node數(shù)組容量大于等于數(shù)組的默認容量,
            // 擴容后的數(shù)組容量值為擴容前的map的容量的2倍,并且擴容后的閾值同樣設(shè)置為擴容前的兩倍,
            // 反之,則只設(shè)置擴容后的容量值為擴容前的map的容量的2倍
            // 這里newCap已經(jīng)在條件里賦值了
            else if ((newCap = oldCap << 1) < MAXIMUM_CAPACITY &&
                     oldCap >= DEFAULT_INITIAL_CAPACITY)
                // C2
                newThr = oldThr << 1; // double threshold
        }
        // 2.擴容前的數(shù)組未初始化并且使用了有參構(gòu)造函數(shù)構(gòu)造
        // 這里在oldCap = 0時執(zhí)行,這里oldThr > 0說明初始化時是有參初始化構(gòu)造的map
        // 自己可以試下無參構(gòu)造函數(shù),threshold的值為0
        // B2
        else if (oldThr > 0) // initial capacity was placed in threshold
            // 使用有參初始化構(gòu)造函數(shù)并且在第一次put操作時會進入執(zhí)行(去看下put源碼)
            // 擴容后的容量大小設(shè)置為原有閾值
            // 例如我上邊的注釋中的例子,這里第一次添加鍵值對時容量設(shè)置為了1024
            newCap = oldThr;
        // 3.擴容前的數(shù)組未初始化并且使用了無參構(gòu)造函數(shù)構(gòu)造
        // B3
        else {               // zero initial threshold signifies using defaults
            // 擴容后的容量 = 默認容量,擴容后的閾值 = 默認容量 * 負載因子
            // 擴容后的容量為16,閾值為12
            newCap = DEFAULT_INITIAL_CAPACITY;
            newThr = (int)(DEFAULT_LOAD_FACTOR * DEFAULT_INITIAL_CAPACITY);
        }
        /**
         * 上邊設(shè)置了新容量和新的閾值,執(zhí)行到這里,你應(yīng)該發(fā)現(xiàn)只有newThr可能沒被賦值,所以這里要繼續(xù)進行一個操作,來對newThr進行賦值
         * 新閾值等于0,照上邊邏輯:
         * 兩種情況:
         * 1.擴容前的node數(shù)組容量有值且擴容后容量超過最大值或者原node數(shù)組容量小于默認初始容量16
         * 2.使用有參構(gòu)造函數(shù),第一次put操作時上邊代碼里沒有設(shè)置newThr
         * D1
         */
        if (newThr == 0) {
            // 應(yīng)該得到的新閾值ft = 新容量 * 負載因子
            float ft = (float)newCap * loadFactor;
            // 假如新容量小于最大容量并且ft小于最大容量則新的閾值設(shè)置為ft,否則設(shè)置成int最大值
            newThr = (newCap < MAXIMUM_CAPACITY && ft < (float)MAXIMUM_CAPACITY ?
                      (int)ft : Integer.MAX_VALUE);
        }
        // 執(zhí)行到這,擴容后的容量和閾值都計算完畢
        // 閾值設(shè)置為新閾值
        threshold = newThr;
        @SuppressWarnings({"rawtypes","unchecked"})
        // 創(chuàng)建擴容后的Node數(shù)組 
        Node[] newTab = (Node[])new Node[newCap];
        // 切換為擴容后的Node數(shù)組,此時還未進行將舊數(shù)組拷貝到新數(shù)組
        table = newTab;
        // E1
        if (oldTab != null) {
            // 原有數(shù)組不為空,將原有數(shù)組數(shù)據(jù)拷貝到新數(shù)組中
            for (int j = 0; j < oldCap; ++j) {
                Node e;
                // 非空元素才進行賦值
                if ((e = oldTab[j]) != null) {
                    // 原有數(shù)值引用置空,方便GC
                    oldTab[j] = null;
                    if (e.next == null)
                        // 桶對應(yīng)的Node只有當前一個節(jié)點,鏈表長度為1
                        // 中括號中計算原有數(shù)組元素在新數(shù)組中存放的位置,
                        // 為什么這么計算?
                        // 正常的想,添加了一個鍵值對,鍵的hash值(當然,這里在HashMap的hash(key)進行了統(tǒng)一處理)
                        // 那么長度是有限的,在這個有限長度下如何放置,類比整數(shù)取余操作,
                        // &操作表明只取e.hash的低n位,n是newCap - 1轉(zhuǎn)換成二進制的有效位數(shù)
                        // 這里記得初始不設(shè)長度時默認16,二進制為10000,減一為1111,低4位
                        // 設(shè)置長度時tableSizeFor重新設(shè)置了長度和16處理類似
                        // 通過&操作所有添加的鍵值對都分配到了數(shù)組中,當然,分配到數(shù)組中同一個位置時會擴展成鏈表或紅黑樹
                        // 添加詳細操作看后邊putVal源碼,這里先不用糾結(jié)
                        newTab[e.hash & (newCap - 1)] = e;
                    else if (e instanceof TreeNode)
                        // 到此說明e.next不為空,那么需要判斷了,
                        // 因為有兩種結(jié)構(gòu),一種是鏈表,一種為紅黑樹
                        // 這里先進行紅黑樹處理,樹的具體處理后邊有時間多帶帶做一章進行說明講解,
                        // 這里先簡單了解,擴容之后,需要對原有的樹進行處理,使得數(shù)據(jù)分散比較均勻。
                        ((TreeNode)e).split(this, newTab, j, oldCap);
                    else { // preserve order
                        /**
                         * 到這里結(jié)合HashMap的結(jié)構(gòu),
                         * 排除上邊兩個條件,這里就進行鏈表結(jié)構(gòu)的處理
                         * 進行鏈表復(fù)制操作
                         * 復(fù)制的時候就有個問題了,舉個例子,原來我是16,現(xiàn)在擴容成了32(原數(shù)組兩倍,我上邊分析里有說明)
                         * 那么我復(fù)制時怎么辦?
                         * 不移動原來的鏈表?
                         * 這里就要想到了我擴容之后訪問的時候不能影響
                         * 那么就需要看下put操作時是怎樣存的,這里先說下,putVal里也可以看到
                         * (n - 1) & hash 和上邊newTab[e.hash & (newCap - 1)] 分析是一樣的
                         * 這里不知道你想到了嗎?擴容之后有什么不同?
                         * 如果還沒什么想法,請繼續(xù)往下看,我等下會說明
                         * 新擴容部分頭尾節(jié)點(hi可以理解成高位)設(shè)置為hiHead,hiTail
                         * 原有部分頭尾節(jié)點(lo可以理解成低位)設(shè)置為loHead,loTail
                         * 這里什么意思呢?
                         * 往下看就好,我下面的注釋詳細說明了為什么定義了兩個鏈表頭尾節(jié)點
                         */
                        Node loHead = null, loTail = null;
                        Node hiHead = null, hiTail = null;
                        Node next;
                        // 這里循環(huán)獲取鏈表元素進行處理
                        do {
                            next = e.next;
                            /**
                             * e.hash & oldCap = 0
                             * 位與操作,這里初學(xué)者要自己寫下多理解下
                             * 舉個例子:
                             * oldCap=32=100000(二進制),newCap=64=1000000(二進制)
                             * 在未擴容之前計算元素所處位置是(oldCap-1) & hash
                             * 全1位與操作,取值范圍落在0~oldCap-1
                             * e.hash & oldCap 只判斷了最高位的那個1位置是否相同
                             * 相同則非0,不同則為0
                             * 為什么要判斷這一位呢?
                             * 我們想一想,擴容之后,計算bucket(桶)位置(即元素落在數(shù)組那個索引位置)時
                             * (newCap-1) & hash和(oldCap-1) & hash兩者對比,只有一位不同
                             * 比如32和64,最高位是1不同,其他位相同
                             * 如果擴容之后最高位為0,則擴容前后得到的bucket位置相同,不需要調(diào)整位置
                             * 如果非0,則是1,則需要將桶位置調(diào)整到更高的索引位置
                             * 而且這里也應(yīng)該明白,同一個bucket下的鏈表(非單一元素)在擴容后
                             * 因為只有一位二進制不同,不是1就是0
                             * 最多分到兩個bucket中,一個是擴容前的bucket(當前所在的bucket),
                             * 一個是擴容后的bucket(新的bucket),
                             * 這里也說明了上邊為什么設(shè)置了兩組頭尾節(jié)點,一組低位鏈表,一組高位鏈表
                             * 擴容前后兩個bucket位置之間差值為原數(shù)組容量值
                             * 上邊32和64,差值為63-31=32=oldCap=10000(二進制)
                             * 所以這下面使用的是oldCap
                             */
                            if ((e.hash & oldCap) == 0) {
                                // 說明當前Node元素位置 = 原數(shù)組中的位置
                                // 放入loHead,loTail這一組中,低位鏈表
                                if (loTail == null)
                                    // 鏈表還未放元素,鏈表頭賦值
                                    loHead = e;
                                else
                                    // 鏈表存在元素,新元素放置在鏈表尾部,next指向新元素
                                    loTail.next = e;
                                // 尾節(jié)點指向改變,變成了新添加的節(jié)點
                                loTail = e;
                            }
                            else {
                                // 類似上邊
                                if (hiTail == null)
                                    hiHead = e;
                                else
                                    hiTail.next = e;
                                hiTail = e;
                            }
                        } while ((e = next) != null);
                        
                        // 上面已經(jīng)處理完了,分成了高低位兩個鏈表,下面就是將這兩個鏈表放置擴容后的新數(shù)組中
                        if (loTail != null) {
                            // 低位鏈表不為空,添加到新數(shù)組,尾節(jié)點next指向置空,因為原有節(jié)點可能還存在next指向
                            loTail.next = null;
                            // 新數(shù)組j處就是原有數(shù)組j處,這里直接將低位首節(jié)點引用賦值給新數(shù)組節(jié)點
                            newTab[j] = loHead;
                        }
                        if (hiTail != null) {
                            hiTail.next = null;
                            // 這里和我上邊注釋分析是一致的,相差的值即為oldCap,即原數(shù)組的容量
                            newTab[j + oldCap] = hiHead;
                        }
                    }
                }
            }
        }
        return newTab;
    }

    /**
     * put操作方法主體
     * hash,key的hash值,上邊講過,HashMap自己處理過的
     * onlyIfAbsent,是否覆蓋原有值,true,不覆蓋原有值
     * evict,LinkedHashMap實現(xiàn)afterNodeInsertion方法時調(diào)用,這里相當于占位符的作用
     */    
    final V putVal(int hash, K key, V value, boolean onlyIfAbsent,
                   boolean evict) {
        Node[] tab; Node p; int n, i;
        // F1
        if ((tab = table) == null || (n = tab.length) == 0)
            // table為空或長度為0時,對table進行初始化,上邊已經(jīng)分析過了
            // 這里也說明了第一次初始化是在這里,而不是使用構(gòu)造方法,排除putMapEntries方式
            n = (tab = resize()).length;
        // 判斷當前需要存儲的鍵值對存放到數(shù)組中的位置是否已經(jīng)存在值(鏈表或者紅黑樹)
        // 即是否已經(jīng)有對應(yīng)key
        // G1
        if ((p = tab[i = (n - 1) & hash]) == null)
            // 不存在,則創(chuàng)建一個新節(jié)點保存
            tab[i] = newNode(hash, key, value, null);
        // G2
        else {
            // 將桶上的值進行匹配,判斷是否存在
            Node e; K k;
            if (p.hash == hash &&
                ((k = p.key) == key || (key != null && key.equals(k))))
                // 鏈表頭節(jié)點(或紅黑樹根節(jié)點)與當前需要保存的hash值相等
                // 并且key值相等,e和p是同一個,說明添加了相同的key
                // e指向p對應(yīng)的節(jié)點
                e = p;
            else if (p instanceof TreeNode)
                // 紅黑樹添加節(jié)點處理,本文不詳細將紅黑樹部分,后面有空會多帶帶抽出講解
                // 返回值可以理解成如果有相同key,則返回對應(yīng)Node,否則返回null(創(chuàng)建了新的Node)
                e = ((TreeNode)p).putTreeVal(this, tab, hash, key, value);
            else {
                // 這里說明非頭節(jié)點(數(shù)組中對應(yīng)的桶的第一個節(jié)點),非紅黑樹結(jié)構(gòu),
                // 說明需要匹配鏈表,判斷鏈表中對應(yīng)的key是否已存在
                // 設(shè)置binCount計算當前桶中bin的數(shù)量,即鏈表長度
                for (int binCount = 0; ; ++binCount) {
                    if ((e = p.next) == null) {
                        // next 為空 無下一個元素 不再繼續(xù)查找 直接新創(chuàng)建直接賦值next
                        p.next = newNode(hash, key, value, null);
                        if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st
                            // 判斷是否樹化,這里就是鏈表樹化條件,在treeifyBin還有個數(shù)組容量判斷,方法也可能只進行擴容操作
                            // 總結(jié)下,即桶中bin數(shù)量大于等于TREEIFY_THRESHOLD=8,數(shù)組容量不能小于MIN_TREEIFY_CAPACITY=64時進行樹化轉(zhuǎn)化
                            // 怎么轉(zhuǎn)成紅黑樹結(jié)構(gòu)這里也不做深入,后續(xù)會進行說明
                            treeifyBin(tab, hash);
                        break;
                    }
                    // 不為空 且節(jié)點為尋找的節(jié)點 終止循環(huán)
                    if (e.hash == hash &&
                        ((k = e.key) == key || (key != null && key.equals(k))))
                        break;
                    p = e;
                }
            }
            // 上邊已經(jīng)檢查完map中是否存在對應(yīng)key的Node節(jié)點,不存在的新創(chuàng)建節(jié)點,這里處理下存在對應(yīng)key的節(jié)點數(shù)據(jù)
            // H1
            if (e != null) { // existing mapping for key
                // 保存下原來的節(jié)點值
                V oldValue = e.value;
                if (!onlyIfAbsent || oldValue == null)
                    // onlyIfAbsent 是否需要覆蓋操作,是則覆蓋
                    e.value = value;
                // 子類實現(xiàn)方法的話可以進行對應(yīng)的后置操作
                afterNodeAccess(e);
                // 返回原值
                return oldValue;
            }
        }
        ++modCount;
        // 實際元素長度,不是容量,是每次添加一個新的鍵值對會加1,覆蓋不增加
        // 判斷是否大于閾值,進行擴容操作
        // I1
        if (++size > threshold)
            resize();
        // 同afterNodeAccess,子類實現(xiàn)方法的話可以進行對應(yīng)的后置操作
        afterNodeInsertion(evict);
        return null;
    }

重點的部分也就是在上面這幾個方法,剩下的源碼部分就不一一貼出來分析了,能看懂我上面說明的部分,基本上除了紅黑樹和jdk1.8的新特性相關(guān)部分,其余部分應(yīng)該基本都能看懂,這里再補充一個序列化方面的問題:

為什么HashMap中的table變量要設(shè)置為transient?在理解這個問題之前,自行去看下序列化代碼writeObject和readObject,然后參考以下鏈接來思考:

https://segmentfault.com/q/10...

HashMap中,由于Entry的存放位置是根據(jù)Key的Hash值來計算,然后存放到數(shù)組中的,對于同一個Key,在不同的JVM實現(xiàn)中計算得出的Hash值可能是不同的。這里不同意思是說我原來在window機器上A是放在Node數(shù)組中0的位置,在Mac上可能是放在Node數(shù)組中5的位置,但是不修改的話,反序列化之后Mac上也是0的位置,這樣導(dǎo)致后續(xù)增加節(jié)點會錯亂,不是我們想要的結(jié)果,故在序列化中HashMap對每個鍵值對的鍵和值序列化,而不是整體,反序列化一個一個取出來,不會造成位置錯亂。

那么JDK1.8中HashMap在多線程環(huán)境下會造成死循環(huán)嗎?

從上邊結(jié)構(gòu)以及處理過程的分析來看,應(yīng)該是不會的,只不過數(shù)據(jù)丟失還是會發(fā)生,這一塊我就不進行驗證了,自行Google,手寫代碼來驗證。同時想多說句,對于一般開發(fā)人員知道HashMap是非線程安全的,多線程情況下使用ConcurrentHashMap即可,后邊有時間ConcurrentHashMap的分析我也會整理出來。

總結(jié)

在重點說明部分我已經(jīng)詳細解釋了resize和put操作的過程,可能有些新人還是不能梳理清楚,我在這里結(jié)合下日常使用總結(jié)下整個過程,方便各位理解:

1.HashMap創(chuàng)建過程(正常狀態(tài)):

2.HashMap resize過程(正常狀態(tài)):

3.HashMap put過程(正常狀態(tài)):

HashMap首先需要理解清楚其內(nèi)部的實現(xiàn)結(jié)構(gòu):數(shù)組+鏈表+紅黑樹,在結(jié)構(gòu)的基礎(chǔ)之上來對源碼進行深入,resize和put操作是最為重要的兩部分,理解了這兩塊,基本上對HashMap的整體處理過程有了一定的認知,另外,一定要自己動手debug,理清數(shù)據(jù)的轉(zhuǎn)換,對了解HashMap有很大的幫助。

文章先從基礎(chǔ)部分說起,解釋了一些名詞,提及了哈希表,從實現(xiàn)結(jié)構(gòu)開始來幫助各位更好的理解源碼操作部分,對重點的幾個部分做出詳細的說明,resize和put操作難點部分也做了相應(yīng)的解釋,希望對各位有所幫助,后邊有空我會將紅黑樹部分的理解分享出來,謝謝。

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

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

相關(guān)文章

  • JDK源碼那些事兒HashMap.TreeNode

    摘要:前言版本以為例是因為之前的紅黑樹操作在文章省略了,這里進行一個解釋,其實源碼里并不是只有這個地方用紅黑樹結(jié)構(gòu),但是總體上都大同小異,故只說明這一部分就好,舉一反三的能力相信各位都應(yīng)該擁有。紅黑樹類型遞歸左右子樹遍歷,直到值相等。 前面幾篇文章已經(jīng)講解過HashMap內(nèi)部實現(xiàn)以及紅黑樹的基礎(chǔ)知識,今天這篇文章就講解之前HashMap中未講解的紅黑樹操作部分,如果沒了解紅黑樹,請去閱讀前面...

    Pandaaa 評論0 收藏0
  • JDK源碼那些事兒之并發(fā)ConcurrentHashMap上篇

    前面已經(jīng)說明了HashMap以及紅黑樹的一些基本知識,對JDK8的HashMap也有了一定的了解,本篇就開始看看并發(fā)包下的ConcurrentHashMap,說實話,還是比較復(fù)雜的,筆者在這里也不會過多深入,源碼層次上了解一些主要流程即可,清楚多線程環(huán)境下整個Map的運作過程就算是很大進步了,更細的底層部分需要時間和精力來研究,暫不深入 前言 jdk版本:1.8 JDK7中,ConcurrentH...

    Leck1e 評論0 收藏0
  • JDK源碼那些事兒之常用ArrayList

    摘要:前面已經(jīng)講解集合中的并且也對其中使用的紅黑樹結(jié)構(gòu)做了對應(yīng)的說明,這次就來看下簡單一些的另一個集合類,也是日常經(jīng)常使用到的,整體來說,算是比較好理解的集合了,一起來看下前言版本類定義繼承了,實現(xiàn)了,提供對數(shù)組隊列的增刪改查操作實現(xiàn)接口,提供隨 前面已經(jīng)講解集合中的HashMap并且也對其中使用的紅黑樹結(jié)構(gòu)做了對應(yīng)的說明,這次就來看下簡單一些的另一個集合類,也是日常經(jīng)常使用到的ArrayL...

    hizengzeng 評論0 收藏0
  • JDK源碼那些事兒之并發(fā)ConcurrentHashMap下篇

    摘要:上一篇文章已經(jīng)就進行了部分說明,介紹了其中涉及的常量和變量的含義,有些部分需要結(jié)合方法源碼來理解,今天這篇文章就繼續(xù)講解并發(fā)前言本文主要介紹中的一些重要方法,結(jié)合上篇文章中的講解部分進行更進一步的介紹回顧下上篇文章,我們應(yīng)該已經(jīng)知道的整體結(jié) 上一篇文章已經(jīng)就ConcurrentHashMap進行了部分說明,介紹了其中涉及的常量和變量的含義,有些部分需要結(jié)合方法源碼來理解,今天這篇文章就...

    Zack 評論0 收藏0
  • JDK源碼那些事兒之紅黑樹基礎(chǔ)下篇

    摘要:強調(diào)一下,紅黑樹中的葉子節(jié)點指的都是節(jié)點。故刪除之后紅黑樹平衡不用調(diào)整。將達到紅黑樹平衡。到此關(guān)于紅黑樹的基礎(chǔ)已經(jīng)介紹完畢,下一章我將就源碼中的進行講解說明,看一看紅黑樹是如何在源碼中實現(xiàn)的。 說到HashMap,就一定要說到紅黑樹,紅黑樹作為一種自平衡二叉查找樹,是一種用途較廣的數(shù)據(jù)結(jié)構(gòu),在jdk1.8中使用紅黑樹提升HashMap的性能,今天就來說一說紅黑樹,上一講已經(jīng)給出插入平衡...

    羅志環(huán) 評論0 收藏0

發(fā)表評論

0條評論

voyagelab

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<