精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

面試必問:hashmap為什么會出現死循環?

新聞
jdk1.7 hashmap的循環依賴問題是面試經常被問到的問題,如何回答不好,可能會被扣分。今天我就帶大家一下梳理一下,這個問題是如何產生的,以及如何解決這個問題。

jdk1.7 hashmap的循環依賴問題是面試經常被問到的問題,如何回答不好,可能會被扣分。今天我就帶大家一下梳理一下,這個問題是如何產生的,以及如何解決這個問題。

一、hashmap的數據結構

先一起看看jdk1.7 hashmap的數據結構

數組 + 鏈表

hashmap會給每個元素的key生成一個hash值,然后根據這個hash值計算一個在數組中的位置i。i不同的元素放在數組的不同位置,i相同的元素放在鏈表上,最新的數據放在鏈表的頭部。

往hashmap中保存元素會調用put方法,獲取元素會調用get方法。接下來,我們重點看看put方法。

二、put方法

重點看看put方法

  1. public V put(K key, V value) { 
  2.     if (table == EMPTY_TABLE) { 
  3.         inflateTable(threshold);    }    if (key == null
  4.         return putForNullKey(value); 
  5.     //根據key獲取hash      
  6.     int hash = hash(key);    //計算在數組中的下表 
  7.     int i = indexFor(hash, table.length);    //變量集合查詢相同key的數據,如果已經存在則更新數據 
  8.     for (Entry<K,V> e = table[i]; e != null; e = e.next) { 
  9.         Object k;        if (e.hash == hash && ((k = e.key) == key || key.equals(k))) { 
  10.             V oldValue = e.value;            e.value = value;            e.recordAccess(this);            //返回已有數據 
  11.             return oldValue; 
  12.         }    }    modCount++;    //如果不存在相同key的元素,則添加新元素 
  13.     addEntry(hash, key, value, i);    return null

再看看addEntry方法

  1. void addEntry(int hash, K key, V value, int bucketIndex) { 
  2.       // 當數組的size >= 擴容閾值,觸發擴容,size大小會在createEnty和removeEntry的時候改變      if ((size >= threshold) && (null != table[bucketIndex])) { 
  3.           // 擴容到2倍大小,后邊會跟進這個方法          resize(2 * table.length);          // 擴容后重新計算hash和index 
  4.           hash = (null != key) ? hash(key) : 0; 
  5.           bucketIndex = indexFor(hash, table.length); 
  6.       }      // 創建一個新的鏈表節點,點進去可以了解到是將新節點添加到了鏈表的頭部      createEntry(hash, key, value, bucketIndex); 
  7.   } 

看看resize是如何擴容的

  1. void resize(int newCapacity) { 
  2.         Entry[] oldTable = table;        int oldCapacity = oldTable.length; 
  3.         if (oldCapacity == MAXIMUM_CAPACITY) { 
  4.             threshold = Integer.MAX_VALUE;            return
  5.         }        // 創建2倍大小的新數組 
  6.         Entry[] newTable = new Entry[newCapacity]; 
  7.         // 將舊數組的鏈表轉移到新數組,就是這個方法導致的hashMap不安全,等下我們進去看一眼 
  8.         transfer(newTable, initHashSeedAsNeeded(newCapacity)); 
  9.         table = newTable; 
  10.         // 重新計算擴容閾值(容量*加載因子) 
  11.         threshold = (int)Math.min(newCapacity * loadFactor, MAXIMUM_CAPACITY + 1); 

出問題的就是這個transfer方法

  1. void transfer(Entry[] newTable, boolean rehash) { 
  2.     int newCapacity = newTable.length;    // 遍歷舊數組 
  3.     for (Entry<K,V> e : table) { 
  4.         // 遍歷鏈表 
  5.         while(null != e) { 
  6.              //獲取下一個元素,記錄到一個臨時變量,以便后面使用 
  7.             Entry<K,V> next = e.next
  8.             if (rehash) { 
  9.                 e.hash = null == e.key ? 0 : hash(e.key); 
  10.             }            // 計算節點在新數組中的下標 
  11.             int i = indexFor(e.hash, newCapacity); 
  12.             // 將舊節點插入到新節點的頭部 
  13.             e.next = newTable[i]; 
  14.             //這行才是真正把數據插入新數組中,前面那行代碼只是設置當前節點的next 
  15.             //這兩行代碼決定了倒序插入 
  16.             //比如:以前同一個位置上是:3,7,后面可能變成了:7、3 
  17.             newTable[i] = e; 
  18.             //將下一個元素賦值給當前元素,以便遍歷下一個元素 
  19.             e = next
  20.         } 
  21.     } 

我來給大家分析一下,為什么這幾個代碼是頭插法,網上很多技術文章都沒有說清楚。

三、頭插法

我們把目光聚焦到這幾行代碼:

  1. //獲取下一個元素,記錄到一個臨時變量,以便后面使用 
  2.   Entry<K,V> next = e.next
  3.   // 計算節點在新數組中的下標  int i = indexFor(e.hash, newCapacity);  // 將舊節點插入到新節點的頭部  e.next = newTable[i]; 
  4.   //這行才是真正把數據插入新數組中,前面那行代碼只是設置當前節點的next 
  5.   newTable[i] = e;  //將下一個元素賦值給當前元素,以便遍歷下一個元素  e = next

假設剛開始hashMap有這些數據

調用put方法需要進行一次擴容,剛開始會創建一個空的數組,大小是以前的2倍,如圖所示:

開始第一輪循環:

  1. //next= 7   e = 3  e.next = 7 
  2. Entry<K,V> next = e.next
  3. // i=3 
  4. int i = indexFor(e.hash, newCapacity);//e.next = null ,剛初始化時新數組的元素為null 
  5. e.next = newTable[i]; 
  6. //給新數組i位置 賦值 3 
  7. newTable[i] = e;// e = 7 
  8. e = next

執行完之后,第一輪循環之后數據變成這樣的

再接著開始第二輪循環:

  1. //next= 5   e = 7  e.next = 5 
  2. Entry<K,V> next = e.next
  3. // i=3 
  4. int i = indexFor(e.hash, newCapacity);//e.next = 3 ,此時相同位置上已經有key=3的值了,將該值賦值給當前元素的next 
  5. e.next = newTable[i]; 
  6. //給新數組i位置 賦值 7 
  7. newTable[i] = e;// e = 5 
  8. e = next

上面會構成一個新鏈表,連接的順序正好反過來了。

由于第二次循環時,節點key=7的元素插到相同位置上已有元素key=3的前面,所以說是采用的頭插法。

四、死循環的產生

接下來重點看看死循環是如何產生的?

假設數據跟元素數據一致,有兩個線程:線程1 和 線程2,同時執行put方法,最后同時調用transfer方法。

線程1 先執行,到 Entry next = e.next; 這一行,被掛起了。

  1. //next= 7   e = 3  e.next = 7 
  2. Entry<K,V> next = e.next
  3. int i = indexFor(e.hash, newCapacity);e.next = newTable[i]; 
  4. newTable[i] = e;e = next

此時線程1 創建的數組會創建一個空數組

接下來,線程2開始執行,由于線程2運氣比較好,沒有被中斷過,執行完畢了。

過一會兒,線程1被恢復了,重新執行代碼。

  1. //next= 7   e = 3  e.next = 7 
  2. Entry<K,V> next = e.next
  3. // i = 3 
  4. int i = indexFor(e.hash, newCapacity);// e.next = null,剛初始化時新數組的元素為null 
  5. e.next = newTable[i]; 
  6. // 給新數組i位置 賦值 3 
  7. newTable[i] = e;// e = 7 
  8. e = next

這時候線程1的數組會變成這樣的

再執行第二輪循環,此時的e=7

  1. //next= 3   e = 7  e.next = 3 
  2. Entry<K,V> next = e.next
  3. // i = 3 
  4. int i = indexFor(e.hash, newCapacity);// e.next = 3,此時相同位置上已經有key=3的值了,將該值賦值給當前元素的next 
  5. e.next = newTable[i]; 
  6. // 給新數組i位置 賦值 7 
  7. newTable[i] = e;// e = 3 
  8. e = next

這里特別要說明的是 此時e=7,而e.next為什么是3呢?

因為hashMap的數據是公共的,還記得線程2中的生成的數據嗎?

此時e=7,那么e.next肯定是3。

經過上面第二輪循環之后,線程1得到的數據如下:

此時由于循環判斷還沒有退出,判斷條件是: while(null != e),所以要開始第三輪循環:

  1. //nextnull   e = 3  e.next = null 
  2. Entry<K,V> next = e.next
  3. // i = 3 
  4. int i = indexFor(e.hash, newCapacity);// e.next = 7,關鍵的一步,由于第二次循環是 key:7 .next = key:3,現在key:3.next = key:7 
  5. e.next = newTable[i]; 
  6. // 給新數組i位置 賦值 3 
  7. newTable[i] = e;// e = nulle = next

由于e=null,此時會退出循環,最終線程1的數據會是這種結構:

key:3 和 key:7又恢復了剛開始的順序,但是他們的next會相互引用,構成環形引用。

注意,此時調用hashmap的get方法獲取數據時,如果只是獲取循環鏈上key:3 和 key:7的數據,是不會有問題的,因為可以找到。就怕獲取循環鏈上沒有的數據,比如:key:11,key:15等,會進入無限循環中導致CPU使用率飆升。

五、如何避免死循環

為了解決這個問題,jdk1.8把擴容是復制元素到新數組由 頭插法 改成了 尾插法 。此外,引入了紅黑樹,提升遍歷節點的效率。在這里我就不過多介紹了,如果有興趣的朋友,可以關注我的公眾號,后面會給大家詳細分析jdk1.8的實現,以及 jdk1.7、jdk1.8 hashmap的區別。

此外,HashMap是非線程安全的,要避免在多線程的環境中使用HashMap,而應該改成使用ConcurrentHashMap。

所以總結一下要避免發生死循環的問題的方法:

  • 改成ConcurrentHashMap

PS. 即使JDK升級到1.8任然有死循環的問題。

責任編輯:未麗燕 來源: 今日頭條
相關推薦

2022-01-18 06:59:50

HashMap循環底層

2022-01-20 08:44:25

HashMap死循環開放性

2013-06-06 13:34:56

HashMap線程不安全

2023-02-03 07:24:49

雙親委派模型

2011-05-17 08:58:29

軟件項目經理

2020-12-17 07:39:30

HashMap死循環數據

2025-01-21 00:00:00

HashMap死循環數據損壞

2023-02-01 07:15:16

2023-02-17 08:02:45

@Autowired@Resource

2021-12-09 12:22:28

MyBatis流程面試

2023-06-07 08:08:43

JVM內存模型

2020-07-28 08:59:22

JavahreadLocal面試

2023-05-15 08:34:36

css浮動

2023-02-15 07:03:41

跨域問題面試安全

2022-06-18 23:10:56

前端模塊循環依賴

2020-03-27 10:08:10

JS異步 I

2021-12-06 11:03:57

JVM性能調優

2021-12-27 08:22:18

Kafka消費模型

2020-10-12 18:00:39

JavaAQS代碼

2020-05-27 12:45:52

HashMapJava加載因子
點贊
收藏

51CTO技術棧公眾號

久久久久久久久久久成人| 在线观看www91| 国产一区二区不卡视频| 成年人免费高清视频| 国产精品免费大片| 欧美一区二区三区视频在线| 六月婷婷在线视频| 北条麻妃在线| 国产成人av电影| 日韩美女毛茸茸| 欧美日韩在线国产| 蜜桃一区二区| 精品国产免费一区二区三区四区 | 欧美视频在线播放| 久久福利一区二区| 国产黄在线播放| 处破女av一区二区| 成人h视频在线| 中文字幕视频网| 欧美1区免费| 亚洲欧美精品一区二区| 高清中文字幕mv的电影| 成人四虎影院| 色婷婷精品大视频在线蜜桃视频 | 国产一区二区三区成人欧美日韩在线观看| 在线精品视频视频中文字幕| 国产精品一区二区在线免费观看| 福利一区二区三区视频在线观看| 欧美视频在线观看免费| 青青视频免费在线观看| 337p日本欧洲亚洲大胆鲁鲁| 久久久久国色av免费看影院| 国产伦精品一区二区三区高清版 | 中文字幕黄色片| 午夜精品av| 日韩在线观看免费av| 瑟瑟视频在线观看| 日本欧美三级| 亚洲国产精品人久久电影| 自拍一级黄色片| 欧美亚洲韩国| 欧美日韩国产页| 霍思燕三级露全乳照| 日本aa在线| 亚洲精品老司机| av磁力番号网| sm国产在线调教视频| 自拍偷拍亚洲欧美日韩| 亚洲国产一区二区三区在线播| 欧美伦理影视网| 26uuu精品一区二区| 久久99国产精品| 午夜视频在线免费播放| av不卡在线观看| 精品免费视频123区| 天天摸夜夜添狠狠添婷婷| 成人一级视频在线观看| 国产伦精品一区二区三区视频孕妇 | 精品国产a毛片| 欧美丰满熟妇bbb久久久| 永久免费精品视频| 亚洲激情视频在线| 一二三不卡视频| 精品日韩在线| 日韩视频免费在线观看| 黄色一级片中国| 欧美日韩mv| 韩剧1988免费观看全集| 国产又大又黄又粗| 日本伊人色综合网| 成人激情av在线| 丁香花免费高清完整在线播放| 成人av在线资源网| 日本一区二区免费看| av电影在线观看网址| 亚洲视频免费在线| 国产亚洲黄色片| 人人鲁人人莫人人爱精品| 在线免费一区三区| 爱情岛论坛亚洲自拍| 精品人人人人| 一本色道久久综合亚洲精品小说 | 免费人成在线观看播放视频| 亚洲男同1069视频| 国产美女无遮挡网站| 国产亚洲一区二区手机在线观看| 欧美日韩国产综合视频在线观看| 91aaa精品| 欧美丝袜美腿| 精品国产网站地址| 精国产品一区二区三区a片| 99在线|亚洲一区二区| 国产精品女人久久久久久| 国产婷婷在线视频| 久久人人爽人人爽| 手机在线视频你懂的| 亚洲精品国产精品国产| 7777精品伊人久久久大香线蕉经典版下载 | 欧美亚洲视频一区二区| 国产一区二区女内射| 成人免费毛片高清视频| 亚洲bbw性色大片| 黄色羞羞视频在线观看| 欧美视频一区二区三区在线观看| 精品人妻一区二区乱码| 精品一区二区三区中文字幕老牛| 久国内精品在线| 亚洲精品一区二三区| 国产91精品一区二区麻豆亚洲| 日本视频一区二区不卡| 51精品在线| 91麻豆精品国产91久久久久久久久 | 久久精品偷拍视频| 白白色亚洲国产精品| 欧美亚洲视频一区| 少妇一区视频| 日韩av一区二区在线| 全程偷拍露脸中年夫妇| 日本不卡中文字幕| 久久婷婷开心| 黑人精品视频| 在线综合+亚洲+欧美中文字幕| 四虎国产精品成人免费入口| 亚洲黄色三级| 91视频婷婷| 高清全集视频免费在线| 欧美午夜电影网| 丰满少妇在线观看资源站| 好看的日韩av电影| 亚洲一区二区三区视频播放| 97电影在线| 91国产成人在线| 欧美一区二区三区成人精品| 亚洲黄色视屏| 国语精品中文字幕| 爱啪视频在线观看视频免费| 日韩欧美在线123| 亚洲天堂黄色片| 国产做a爰片久久毛片| 一本色道婷婷久久欧美 | 91免费福利视频| 色综合久久影院| 欧美日韩一级片在线观看| 久久亚洲AV无码专区成人国产| 亚洲自啪免费| 久久66热这里只有精品| 丝袜诱惑一区二区| 国产丝袜一区视频在线观看 | 免费在线亚洲| 久久久久久国产精品一区| 国产社区精品视频| 日韩精品亚洲元码| 欧美精品韩国精品| 久久久噜噜噜久久中文字幕色伊伊| 欧美日韩成人免费视频| 妖精一区二区三区精品视频| 国产va免费精品高清在线观看| 青青草手机在线| 在线观看亚洲专区| 色www亚洲国产阿娇yao| 美国三级日本三级久久99| 中文字幕在线中文字幕日亚韩一区 | 电影在线高清| 欧美人体做爰大胆视频| 精品国产精品国产精品| 成人丝袜18视频在线观看| 欧美成人高潮一二区在线看| 牲欧美videos精品| 国产精品自在线| av观看在线| 亚洲电影免费观看高清完整版在线 | 中文字幕人成人乱码| 国产成人看片| 日韩一区二区三区免费| 日韩中文字幕久久| 性欧美一区二区三区| 午夜天堂影视香蕉久久| 第一次破处视频| 国产一区二区精品久久| 玩弄中年熟妇正在播放| 日韩欧美1区| 国产日产精品一区二区三区四区 | 91国产免费观看| 久久综合九色综合88i| 男人天堂网站在线| 亚洲一区三区视频在线观看| 国产精品播放| 日本一区视频在线播放| 女同性恋一区二区| 3d动漫一区二区三区| 国产免费中文字幕| 亚洲专区区免费| 午夜精品久久久久久久蜜桃| 日本学生初尝黑人巨免费视频| 日日噜噜夜夜狠狠久久波多野| 亚欧美无遮挡hd高清在线视频| 亚洲自拍高清视频网站| 91九色国产在线播放| 国产一区二区三区欧美| 亚洲黄色小说网址| 在线观看欧美日本| 国产精品成人国产乱| 国产精品电影一区二区三区| v天堂中文在线| 国产高清视频一区| 欧在线一二三四区| 亚洲午夜视频| 男人的天堂成人| 婷婷激情久久| 成人做爰66片免费看网站| free欧美| 97视频com| jizz性欧美10| 中文日韩电影网站| 午夜视频免费看| 精品国产一区二区三区久久影院| 真实的国产乱xxxx在线91| 香蕉乱码成人久久天堂爱免费| 99久久99久久精品国产| 国产三级欧美三级日产三级99 | 久久国产生活片100| 男人用嘴添女人下身免费视频| 亚洲一区色图| 亚洲一区二区三区在线观看视频 | 国产91对白在线观看九色| 91制片厂毛片| 日本亚洲欧美天堂免费| 亚洲中文字幕无码中文字| 精品成人久久| avav在线播放| 狠狠综合久久| 精品久久久无码人妻字幂| 希岛爱理av一区二区三区| 亚洲精品国产精品国自产| 欧美日韩水蜜桃| 日韩av电影免费在线| 亚洲欧洲色图| 蜜桃av噜噜一区二区三| 亚洲精品中文字幕99999| 精品国产免费久久久久久尖叫| 一区二区中文字幕在线观看| 91视频网页| 成人爽a毛片免费啪啪红桃视频| 亚洲一区二区三区四区在线播放| 成人国产综合| 成人中文字幕+乱码+中文字幕| 欧洲午夜精品| 91在线视频一区| 精品三级国产| 爱情岛论坛亚洲入口| a看欧美黄色女同性恋| 国产91aaa| 老汉色老汉首页av亚洲| 精品久久久久久中文字幕动漫| 琪琪久久久久日韩精品| 久久影视中文粉嫩av| 精品在线99| 日日骚一区二区网站| 成人高清电影网站| 福利网在线观看| 欧美午夜国产| 91猫先生在线| 蜜桃久久精品一区二区| 亚洲激情在线看| 国产成人亚洲综合a∨婷婷图片| 26uuu国产| 久久理论电影网| 一级特黄曰皮片视频| 亚洲欧洲国产日韩| 久草网在线观看| 色综合久久久久| 夜夜狠狠擅视频| 精品成人一区二区| 欧美xxx.com| 久久精品国产精品| 91豆花视频在线播放| 日本久久久久久久久久久| 欧洲亚洲精品| 国产综合av一区二区三区| 欧美亚洲国产精品久久| 粉嫩av一区二区三区天美传媒 | 日本一区二区三区在线播放| 成人精品高清在线视频| 999久久久| 国产一区二区三区站长工具| 免费观看黄色的网站| 国产亚洲午夜| 亚洲理论中文字幕| 91网站最新网址| 成年人一级黄色片| 欧美性videos高清精品| 国产精品免费无遮挡| 国产丝袜精品视频| 污污影院在线观看| 国产成人在线视频| 亚洲高清在线一区| 欧洲久久久久久| 韩日成人在线| 91pony九色| 久久久精品中文字幕麻豆发布| 欧美偷拍第一页| 91传媒视频在线播放| 亚洲国产一二三区| www.色综合| 日韩和的一区二在线| 韩国成人一区| 欧美日本不卡高清| 狠狠躁狠狠躁视频专区| 91社区在线播放| 国产亚洲精久久久久久无码77777| 91国偷自产一区二区开放时间| 丰满少妇高潮在线观看| 日韩有码在线观看| 欧美xo影院| 噜噜噜噜噜久久久久久91| 国产精品红桃| 九九热视频免费| 国产精品丝袜在线| 潘金莲一级淫片aaaaaa播放| 精品成人一区二区三区四区| 国产一二区在线| 国产欧美 在线欧美| 国内成人精品| 无遮挡又爽又刺激的视频| av电影在线观看完整版一区二区| 日本aⅴ在线观看| 欧美人牲a欧美精品| 999国产在线视频| 国产精品成人v| 九九亚洲视频| 成人av一级片| 久久综合色天天久久综合图片| 日本网站免费观看| 精品国产区一区| 99在线视频影院| 国产一区不卡在线观看| 最新亚洲一区| 日本一区二区在线免费观看| 亚洲丰满少妇videoshd| www.xxxx国产| 欧美黑人xxx| 91麻豆精品国产91久久久久推荐资源| 国产在线观看欧美| 国产jizzjizz一区二区| 国产午夜精品无码| 亚洲福利视频二区| 国产不卡人人| 欧美日韩电影一区二区| 久久精品官网| 国产农村妇女精品一区| 精品视频在线视频| 欧美三级电影一区二区三区| 国产一区二区在线免费| 91精品二区| 欧美老女人bb| 五月激情综合网| 欧美人体大胆444www| 国产精品都在这里| 99精品视频在线| 人妻精油按摩bd高清中文字幕| 亚洲一区二区四区蜜桃| 艳母动漫在线看| 国产精品久久久久久久久| 日韩精品欧美| 女人扒开双腿让男人捅 | 欧美理论电影在线播放| 精品精品国产三级a∨在线| 欧美视频第一区| 国产精品久久久久四虎| 国产精品综合在线| 欧美精品激情blacked18| 亚洲8888| 日韩av加勒比| 黑人巨大精品欧美一区二区一视频 | 白嫩白嫩国产精品| 日批视频在线免费看| 国产精品久久久久久久久果冻传媒 | av不卡免费电影| 中国a一片一级一片| 欧美精品在线免费| 亚洲老女人视频免费| aaa一级黄色片| 狠狠色狠狠色综合日日小说| 1区2区3区在线观看| 国产一区二区久久久| 日本人妖一区二区| 久久久久无码国产精品| 亚洲午夜精品久久久久久性色| 日韩精品免费视频一区二区三区| 国产免费黄视频| 日韩一区中文字幕| 日本护士...精品国| 亚洲a成v人在线观看| 亚洲欧美日本视频在线观看| 日本少妇aaa| 精品视频在线播放| 亚洲国产精品免费视频| 污版视频在线观看| 岛国av一区二区三区| 国产高清一区二区三区视频|