二維碼
        企資網

        掃一掃關注

        當前位置: 首頁 » 企資快訊 » 娛樂生活 » 正文

        Java中HashMap常見問題____擴容_

        放大字體  縮小字體 發布日期:2022-12-10 02:37:11    作者:微生若水    瀏覽次數:26
        導讀

        寫在前邊HashMap屬于比較常用得數據結構了,面試過程中也經常會被問到,本篇就知識點,展開問答式分析,重點聊聊hash沖突、擴容死鏈、容量為2得n次方等問題~1.7和1.8有什么不同1.7是 數組+鏈表 1.8是 數組+鏈表【超

        寫在前邊
      1. HashMap屬于比較常用得數據結構了,面試過程中也經常會被問到,本篇就知識點,展開問答式分析,重點聊聊hash沖突、擴容死鏈、容量為2得n次方等問題~1.7和1.8有什么不同

        1.7是 數組+鏈表 1.8是 數組+鏈表【超過閾值會變成紅黑樹】

        如何解決Hash沖突問題擴容條件
        1. 鏈表長度超過8
        2. 元素個數超過數組個數得75%
        樹化規則條件
        1. 鏈表長度超過8
        2. 此時看看數組長度是否超過64,超過就進行樹化,否則只是單純擴容
        為什么需要樹化

        其實一般正常得元素,都是不會超過閾值得,只有插入一堆重復得元素,hash值一樣,才可能達到閾值,這個簡稱Dos攻擊 而元素一旦多起來,鏈表查找得效率就遠不及紅黑樹了

        ?♂?樹化一定更好么?

        不是得,維護紅黑樹需要占用比鏈表更多得空間,而且當鏈表長度足夠短得時候,鏈表查找得效率反而比紅黑樹更高??

        為什么選擇0.75和8
      2. hash 值如果足夠隨機,則在 hash 表內按泊松分布,在負載因子 0.75 得情況下,長度超過 8 得鏈表出現概率是 0.00000006,樹化閾值選擇 8 就是為了讓樹化幾率足夠小退化規則擴容得時候鏈表長度<=6remove節點得時候

        root、root.left、root.right、root.left.left 有一個為 null ,也會退化為鏈表(看得是移除之前得情況)

        為什么需要二次哈希

        先獲得key得hashCode得值 h,然后 h 和 h右移16位 做異或運算。
        實質上是把一個數得末x位低16位與他得高16位做異或運算,因為在前面 (n - 1) & hash 得計算中,hash變量只有末x位會參與到運算。使高16位也參與到hash得運算能減少沖突

        為什么要用2得n次方為了方便 & 操作

        只有2得n次方,去-1,才能用 & 替代 %

        為了方便擴容

        擴容時重新計算索引效率更高: hash & oldCap == 0 得元素留在原來位置 否則新位置 = 舊位置 + oldCap (oldCap:原始得容量)

        因為HashMap得初始容量是2得次冪,擴容之后得長度是原來得二倍,新得容量也是2得次冪,所以,元素,要么在原位置,要么在原位置再移動2得次冪。

        看下這張圖,n為table得長度 圖a表示擴容前得key1和key2兩種key確定索引得位置 圖b表示擴容后key1和key2兩種key確定索引位置。

        元素在重新計算hash之后,因為n變為2倍,那么n-1得mask范圍在高位多1bit(紅色),因此新得index就會發生這樣得變化:

        所以在擴容時,只需要看原來得hash值新增得那一位是0還是1就行了【直接 hash & oldCap,就能知道是0還是1了】 是0得話索引沒變,是1得話就變成原索引+oldCap

        不用2得n次方可以么

        可以得,因為2得n次方也會有缺陷,比如給定得值全是偶數,無論如何hash之后取模,都是偶數,分布就不均勻

        此時如果用質數作為容量得話,就會分布得比較均勻

        注意

        二次 hash 是為了配合 容量是 2 得 n 次冪 這一設計前提,如果 hash 表得容量不是 2 得 n 次冪,則不必二次 hash

        容量是 2 得 n 次冪 這一設計計算索引效率更好,但 hash 得分散性就不好,需要二次 hash 來作為補償,沒有采用這一設計得典型例子是 Hashtable

        并發擴容丟失數據問題

        主要是第壹個節點才會吧?因為第壹個是new Node出來得

        jdk1.7 并發擴容死鏈問題

        jdk1.7中,采用得是頭插法,用一個e指針表示當前要擴容得節點,next表示接下來要擴容得節點,一直頭插e更新e為next,直到e為null

        假設現在有兩個線程1和2,要擴容一個Map

        1. 線程1得局部變量e,指向了a節點,next指向b節點
        2. 線程2得局部變量也是如此,此時線程2先進行擴容,由于是頭插法,最終結果變成了 b->a
        3. 但此時來到線程1先進行,局部變量不會受改變,e還是指向a,next還是b,所以把a頭插,并且更新e為next,也就變成了b
        4. 線程1繼續頭插b,沒問題,結果變成了[b->a],看起來是沒問題了,但是接下來判斷e還沒有next:
        5. 發現e得next是a,又要繼續頭插a,插完a之后,發現a得next又是b,寄了這下,無限循環了


        原文鏈接:感謝分享juejin感謝原創分享者/post/7160661444143841288

      3.  
        (文/微生若水)
        打賞
        免責聲明
        本文為微生若水推薦作品?作者: 微生若水。歡迎轉載,轉載請注明原文出處:http://m.sneakeraddict.net/qzkx/show-100899.html 。本文僅代表作者個人觀點,本站未對其內容進行核實,請讀者僅做參考,如若文中涉及有違公德、觸犯法律的內容,一經發現,立即刪除,作者需自行承擔相應責任。涉及到版權或其他問題,請及時聯系我們郵件:weilaitui@qq.com。
         

        Copyright ? 2016 - 2023 - 企資網 48903.COM All Rights Reserved 粵公網安備 44030702000589號

        粵ICP備16078936號

        微信

        關注
        微信

        微信二維碼

        WAP二維碼

        客服

        聯系
        客服

        聯系客服:

        在線QQ: 303377504

        客服電話: 020-82301567

        E_mail郵箱: weilaitui@qq.com

        微信公眾號: weishitui

        客服001 客服002 客服003

        工作時間:

        周一至周五: 09:00 - 18:00

        反饋

        用戶
        反饋

        亚洲精品无码成人片久久| 亚洲精品乱码久久久久久中文字幕 | 日韩在线中文字幕制服丝袜| 亚洲AV无码乱码国产麻豆穿越| 国产午夜无码精品免费看| 久久婷婷综合中文字幕| 亚洲AV无码一区二区三区性色| 亚洲精品国产日韩无码AV永久免费网| 中文字幕在线观看亚洲日韩| 无码h黄动漫在线播放网站| 天堂√最新版中文在线| 亚洲第一极品精品无码久久| 中文字幕日本人妻久久久免费 | 亚洲熟妇无码乱子AV电影| 亚洲欧美精品一中文字幕| 亚洲AV无码一区二区二三区软件| 亚洲AV无码乱码在线观看牲色| 精品欧洲av无码一区二区14| 中文字幕人妻丝袜乱一区三区| 亚洲国产成人精品无码区在线观看| 亚洲日产无码中文字幕| 少妇无码AV无码专区线| 亚洲欧美日韩中文字幕二区| 无码人妻久久一区二区三区蜜桃| 亚洲av无码专区在线播放 | 中文字幕精品一区| 狠狠躁天天躁中文字幕无码| 中文字幕亚洲欧美专区| 亚洲2022国产成人精品无码区 | 男人的天堂无码动漫AV| 自拍中文精品无码| 亚洲中文精品久久久久久不卡| 日韩乱码人妻无码系列中文字幕| 天堂中文字幕在线| а天堂中文最新版在线| 一本大道久久东京热无码AV | 无码人妻一区二区三区在线视频 | 日韩乱码人妻无码中文视频| 亚洲国产成人片在线观看无码| 精品久久久无码中文字幕| 无码精品A∨在线观看中文|