欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java源碼解析HashMap的resize函數(shù)

 更新時間:2019年01月05日 15:35:41   作者:李燦輝  
今天小編就為大家分享一篇關于Java源碼解析HashMap的resize函數(shù),小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧

HashMap的resize函數(shù),用于對HashMap初始化或者擴容。

首先看一下該函數(shù)的注釋,如下圖。從注釋中可以看到,該函數(shù)的作用是初始化或者使table的size翻倍。如果table是null,那么就申請空間進行初始化。否則,因為我們在使用2的指數(shù)的擴張,在原來table的每個位置的元素,在新的table中,他們要么待在原來的位置,要么移動2的指數(shù)的偏移。從這里可以看出,擴容前table每個位置上如果有多個元素,元素之間組成鏈表時,在擴容后,該鏈表中的元素,有一部分會待在原地,剩下的元素會往后移動2的指數(shù)的偏移。

  /**
   * Initializes or doubles table size. If null, allocates in
   * accord with initial capacity target held in field threshold.
   * Otherwise, because we are using power-of-two expansion, the
   * elements from each bin must either stay at same index, or move
   * with a power of two offset in the new table.
   * @return the table
   **/

接下來看一下resize的代碼,如下

  final Node<K,V>[] resize() {
    Node<K,V>[] oldTab = table;
    int oldCap = (oldTab == null) ? 0 : oldTab.length;
    int oldThr = threshold;
    int newCap, newThr = 0;
    if (oldCap > 0) {
      if (oldCap >= MAXIMUM_CAPACITY) {
        threshold = Integer.MAX_VALUE;
        return oldTab;
      }
      else if ((newCap = oldCap << 1) < MAXIMUM_CAPACITY &&
           oldCap >= DEFAULT_INITIAL_CAPACITY)
        newThr = oldThr << 1; // double threshold
    }
    else if (oldThr > 0) // initial capacity was placed in threshold
      newCap = oldThr;
    else {        // zero initial threshold signifies using defaults
      newCap = DEFAULT_INITIAL_CAPACITY;
      newThr = (int)(DEFAULT_LOAD_FACTOR * DEFAULT_INITIAL_CAPACITY);
    }
    if (newThr == 0) {
      float ft = (float)newCap * loadFactor;
      newThr = (newCap < MAXIMUM_CAPACITY && ft < (float)MAXIMUM_CAPACITY ?
           (int)ft : Integer.MAX_VALUE);
    }
    threshold = newThr;
    @SuppressWarnings({"rawtypes","unchecked"})
      Node<K,V>[] newTab = (Node<K,V>[])new Node[newCap];
    table = newTab;
    if (oldTab != null) {
      for (int j = 0; j < oldCap; ++j) {
        Node<K,V> e;
        if ((e = oldTab[j]) != null) {
          oldTab[j] = null;
          if (e.next == null)
            newTab[e.hash & (newCap - 1)] = e;
          else if (e instanceof TreeNode)
            ((TreeNode<K,V>)e).split(this, newTab, j, oldCap);
          else { // preserve order
            Node<K,V> loHead = null, loTail = null;
            Node<K,V> hiHead = null, hiTail = null;
            Node<K,V> next;
            do {
              next = e.next;
              if ((e.hash & oldCap) == 0) {
                if (loTail == null)
                  loHead = e;
                else
                  loTail.next = e;
                loTail = e;
              }
              else {
                if (hiTail == null)
                  hiHead = e;
                else
                  hiTail.next = e;
                hiTail = e;
              }
            } while ((e = next) != null);
            if (loTail != null) {
              loTail.next = null;
              newTab[j] = loHead;
            }
            if (hiTail != null) {
              hiTail.next = null;
              newTab[j + oldCap] = hiHead;
            }
          }
        }
      }
    }
    return newTab;
  }

擴容的過程分為兩部分,第一部分是對threshold和table的初始化或者重新計算,第二部分是對HashMap中的元素進行重新放置。初始化的過程比較簡單,基本就是使用默認值,初始化HashMap的各個成員變量。重新計算時,是會申請一個2倍大小的Node數(shù)組,用作的新的HashMap的存儲空間。

之后的過程是,對原來HashMap中的每一個位置進行遍歷,把該位置上的各個元素重新放置到新的table中。所以在循環(huán)的過程中,會定義loHead,loTail,hiHead,hiTail,分別表示留著原地的鏈表的頭和尾,移動到更高位置的鏈表的頭和尾。這里需要注意一點,在jdk1.8中,擴容后鏈表中元素的順序和擴容前鏈表中元素的位置,是相同的,并不會像jdk1.7那樣會發(fā)生逆序。

總結

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對腳本之家的支持。如果你想了解更多相關內(nèi)容請查看下面相關鏈接

相關文章

  • springboot之自動裝配全過程

    springboot之自動裝配全過程

    這篇文章主要介紹了springboot之自動裝配全過程,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-08-08
  • Java中的緩沖流詳細解析

    Java中的緩沖流詳細解析

    這篇文章主要介紹了Java中的緩沖流詳細解析,緩沖流可以分為字節(jié)緩沖流,字符緩沖流,字節(jié)緩沖流可分為字節(jié)輸?入緩沖流,字節(jié)輸出緩沖流,字符緩沖流可以分為字符輸入緩沖流,字符輸出緩沖流,需要的朋友可以參考下
    2023-11-11
  • java實現(xiàn)動態(tài)數(shù)組

    java實現(xiàn)動態(tài)數(shù)組

    這篇文章主要為大家詳細介紹了java實現(xiàn)動態(tài)數(shù)組,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • Java實現(xiàn)SMS短信通發(fā)送手機驗證碼案例講解

    Java實現(xiàn)SMS短信通發(fā)送手機驗證碼案例講解

    這篇文章主要介紹了Java實現(xiàn)SMS短信通發(fā)送手機驗證碼案例講解,本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下
    2021-08-08
  • 一文帶你深入剖析Java線程池的前世今生

    一文帶你深入剖析Java線程池的前世今生

    這篇文章主要帶大家介紹了深入剖析一下Java線程池的前世今生,了解線程池的原理以及為什么需要線程池。文中的示例代碼講解詳細,需要的可以參考一下
    2022-10-10
  • Spring Boot利用@Async如何實現(xiàn)異步調(diào)用:自定義線程池

    Spring Boot利用@Async如何實現(xiàn)異步調(diào)用:自定義線程池

    這篇文章主要給大家介紹了關于Spring Boot利用@Async如何實現(xiàn)異步調(diào)用:自定義線程池的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2018-05-05
  • mybatis-plus排除非表中字段的操作

    mybatis-plus排除非表中字段的操作

    這篇文章主要介紹了mybatis-plus排除非表中字段的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • 詳解Java的Struts框架中棧值和OGNL的使用

    詳解Java的Struts框架中棧值和OGNL的使用

    這篇文章主要介紹了Java的Struts框架中棧值和OGNL的使用,Struts框架是Java的SSH三大web開發(fā)框架之一,需要的朋友可以參考下
    2015-11-11
  • Java從零實現(xiàn)超市會員管理系統(tǒng)

    Java從零實現(xiàn)超市會員管理系統(tǒng)

    這篇文章主要為大家詳細介紹了Java實現(xiàn)超市會員管理系統(tǒng),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-12-12
  • Java程序生成Access文件代碼實例

    Java程序生成Access文件代碼實例

    這篇文章主要介紹了Java程序生成Access文件代碼實例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2019-09-09

最新評論