HashMap添加元素的流程圖

文章目錄

    • JDK7 vs JDK8 的 HashMap 結構變化
    • Java8 中哈希表的紅黑樹優化機制
    • HashMap 添加元素的完整流程解析
      • 1. 計算 key 的哈希值并確定索引
      • 2. 檢查該索引位置是否已有元素
      • 3. 處理哈希沖突
      • 4. 判斷當前存儲結構(鏈表還是紅黑樹)
      • 5. 判斷鏈表長度是否超過 8
      • 6. 觸發擴容的判斷
      • 7. 進行擴容
      • 8. 遷移元素
    • 流程圖
    • 補充
      • HashMap 的初始容量與負載因子優化

JDK7 vs JDK8 的 HashMap 結構變化

在 JDK7 及更早版本,HashMap 采用 數組 + 鏈表 結構,當哈希沖突較多時,鏈表可能變得很長,導致查詢性能從 O(1) 退化到 O(n)。

在 JDK8,為了優化鏈表查詢性能,引入了紅黑樹

  • 仍然采用 數組 作為底層存儲。
  • 當某個桶中的鏈表長度 超過 8,并且 table 的大小 ≥ 64 時,鏈表轉換為 紅黑樹
  • 這樣,在極端情況下,查詢性能從 O(n) 降低到O(log n)

Java8 中哈希表的紅黑樹優化機制

從JDK8開始,為了優化哈希沖突情況下的查找性能,當哈希桶中的鏈表長度超過 8 時,鏈表會轉換為紅黑樹。紅黑樹是一種自平衡二叉搜索樹,將最壞情況下的查找復雜度從 O(n) 降低到 O(log n)。(如果沒引入紅黑樹,則最壞查找復雜度是O(n))

當樹中元素數量低于 6 時,紅黑樹會退化回鏈表,以減少不必要的樹操作開銷,提高小規模數據場景下的性能。

HashMap 添加元素的完整流程解析

1. 計算 key 的哈希值并確定索引

  • 通過 key.hashCode() 計算出哈希值。
  • 采用 (哈希值 & (table.length - 1)) 計算索引,以確定 key 應該存放在 table 數組中的那個位置。

2. 檢查該索引位置是否已有元素

  • 如果該索引位置為空table[index] == null),說明當前 key 沒有發生哈希沖突,直接存入該位置,并檢查是否需要擴容。【在 HashMap 里,負載因子(loadFactor)默認值是 0.75,表示當元素個數達到 容量 * 0.75 時,就會觸發擴容】
  • 如果該索引位置已有元素,說明發生了哈希沖突,進入下一步處理。

3. 處理哈希沖突

在索引位置已有元素的情況下,需要判斷該 key 是否已經存在:

  • 如果 key 與已有節點的 key 相同,說明是更新操作,直接替換 value
  • 如果 key 不同,說明該索引位置是個鏈表或紅黑樹,需要進一步處理。

4. 判斷當前存儲結構(鏈表還是紅黑樹)

  • 如果該索引處存儲的是紅黑樹,按照紅黑樹的插入規則執行插入操作,流程結束。
  • 如果是鏈表,則遍歷鏈表:
    • 如果鏈表中存在相同的 key,則更新 value,流程結束。
    • 如果鏈表中沒有相同 key,則在鏈表末尾插入新節點,并繼續下一步處理。

5. 判斷鏈表長度是否超過 8

  • 如果鏈表長度 ≤ 8,不做額外處理。
  • 如果鏈表長度 > 8,則需要判斷是否轉換為紅黑樹:
    • 如果 table 長度 >= 64,則將鏈表轉換為紅黑樹,流程結束。
    • 如果 table 長度 < 64,則不轉換為紅黑樹,而是觸發擴容(見步驟 7)。

6. 觸發擴容的判斷

在完成插入后,需要判斷是否需要擴容:

  • size >= 閾值(threshold = table.length * 0.75) 時觸發擴容。
  • 擴容是基于整個 HashMap 的大小,而不是單個鏈表的長度,即使單個鏈表過長,也不會單獨擴容,而是考慮整體 size

7. 進行擴容

  • 擴容策略table 容量翻倍(如 16 → 32,32 → 64)。
  • 調整擴容閾值:新閾值 = 新容量 * 0.75
  • 重新計算 key 的索引
    • 由于 table.length 發生變化,所有 key 需要重新計算索引并遷移到新的 table
    • HashMap 采用高位 & 低位拆分的方式優化了 rehash 過程,使得某些 key 的新索引保持不變,而另一些 key 需要移動到新位置。

8. 遷移元素

  • table 的數據逐個遷移到新 table
  • 遷移規則
    • 計算 oldIndex = hash & (oldCapacity - 1)newIndex = hash & (newCapacity - 1)
    • 低位不變,高位索引變化,減少數據遷移的計算量。

流程圖

在這里插入圖片描述


補充

HashMap 的初始容量與負載因子優化

HashMap 的默認初始容量為 16負載因子為 0.75,這一組合在性能與空間之間取得了平衡。較高的負載因子(如 1.0)可減少空間浪費,但會增加哈希沖突的概率;較低的負載因子則減少哈希沖突,但會增加內存開銷。 (如果可預估 HashMap 的存儲需求,建議提前設置合適的初始容量,以減少動態擴容帶來的性能損耗。)

?覺得有用的可以留個關注~~?

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/898861.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/898861.shtml
英文地址,請注明出處:http://en.pswp.cn/news/898861.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Excel(進階篇):powerquery詳解、PowerQuery的各種用法,逆透視表格、雙行表頭如何制作透視表、不規則數據如何制作數據透視表

目錄 PowerQuery工具基礎修改現有數據理規則PowerQuery抓取數據的兩種方式多文件合并透視不同表結構多表追加數據透視追加與合并整理橫向表格:逆透視 數據用拆分工具整理數據算賬齡 不等步長值組合合并文件夾中所有文件PowerQuery處理CSV文件雙行表頭、帶合并單元格如何做數據…

從零開始:使用 Cython + JNI 在 Android 上運行 Python 算法

1. 引言 在 Android 設備上運行 Python 代碼通常面臨性能、兼容性和封裝等挑戰。尤其是當你希望在 Android 應用中使用 Python 編寫的計算密集型算法時&#xff0c;直接運行 Python 代碼可能導致較高的 CPU 占用和較差的性能。為了解決這個問題&#xff0c;我們可以使用 Cytho…

請為下面的html添加一個修改按鈕,以便對書名、價格進行修改

下面的HTML段落&#xff0c;在書名和價格輸入錯誤的情況下&#xff0c;無法進行修改。添加一個按鈕&#xff0c;對已經輸入的信息進行修改。 <!DOCTYPE html> <html> <head> <meta charset"utf-8"> <title></…

FFmpeg + ?Qt? 簡單視頻播放器代碼

一個基于 ?FFmpeg 4.x? 和 ?Qt? 的簡單視頻播放器代碼示例&#xff0c;實現視頻解碼和渲染到 Qt 窗口的功能。 1&#xff09;ffmpeg庫界面&#xff0c;視頻解碼支持軟解和硬解方式。 2&#xff09;QImage/QPixmap顯示視頻圖片。 ?1. Qt 項目配置&#xff08;.pro 文件&…

如何在百度搜索上刪除與自己名字相關的資料

個人信息的網絡足跡如同一張無形的網&#xff0c;將我們與世界的每一個角落緊密相連。然而&#xff0c;當某些與自己名字相關的資料不再希望被公眾輕易檢索到時&#xff0c;如何在百度搜索中有效“隱身”&#xff0c;成為了一個亟待解決的問題。面對復雜多變的網絡環境&#xf…

WebSocket:現代實時通信協議的深度解析與實踐

一、背景與演進歷程 1.1 傳統實時通信的困境 // 典型的HTTP輪詢偽代碼 while(true) {auto response http_client.get("/messages");if(response.has_data()) process(response);std::this_thread::sleep_for(1s); // 固定間隔輪詢 } 高延遲&#xff1a;輪詢間隔導…

[貪心算法]最長回文串 增減字符串匹配 分發餅干

1.最長回文串 我們可以存下每個字母的個數&#xff0c;然后分類討論 如果是奇數就減一加到結果中如果是偶數就直接加入即可 最后判斷長度跟原字符串的差距&#xff0c;如果小于原數組說明有奇數結果1 class Solution { public:int longestPalindrome(string s) {int ret0;//1.計…

STM32 的tf卡驅動

基于STM32的TF卡驅動的基本實現步驟和相關代碼示例,主要使用SPI接口來與TF卡進行通信。 硬件連接 將TF卡的SPI接口與STM32的SPI引腳連接,通常需要連接SCK(時鐘)、MOSI(主出從入)、MISO(主入從出)和CS(片選)引腳。 軟件實現 初始化SPI 配置SPI的工作模式、時鐘頻率…

目標檢測中的非極大值抑制(NMS)原理與實現解析

一、技術背景 在目標檢測任務中&#xff0c;模型通常會對同一目標生成多個重疊的候選框&#xff08;如錨框或預測框&#xff09;。非極大值抑制&#xff08;Non-Maximum Suppression, NMS&#xff09; 是一種關鍵的后處理技術&#xff0c;用于去除冗余的檢測結果&#xff0c;保…

探秘鴻蒙 HarmonyOS NEXT:鴻蒙存儲核心技術全解析

引言 本文章基于HarmonyOS NEXT操作系統&#xff0c;API12以上的版本。 在 ArkTS (ArkUI 框架) 中&#xff0c;用戶首選項 (Preferences) 和 持久化存儲 (PersistentStorage) 都用于數據存儲&#xff0c;但它們有不同的應用場景和特點。 1. 用戶首選項 (Preferences) 概念&a…

Leetcode—15. 三數之和(哈希表—基礎算法)

題目&#xff1a; 給你一個整數數組 nums &#xff0c;判斷是否存在三元組 [nums[i], nums[j], nums[k]] 滿足 i ! j、i ! k 且 j ! k &#xff0c;同時還滿足 nums[i] nums[j] nums[k] 0 。請你返回所有和為 0 且不重復的三元組。 注意&#xff1a;答案中不可以包含重復的…

Linux 啟動Jar腳本設置開機自啟【超級詳細】

Linux 啟動Jar腳本&&設置開機自啟【超級詳細】 概要服務器開機自啟服務重啟腳本 概要 最近在Linux服務器中部署了一個項目&#xff08;單機版&#xff09;&#xff0c;每次更新服務的時候需要用到好幾個命令&#xff0c;停止服務&#xff0c;再重啟&#xff0c;并且服…

【第21節】windows sdk編程:網絡編程基礎

目錄 引言&#xff1a;網絡編程基礎 一、socket介紹(套接字) 1.1 Berkeley Socket套接字 1.2 WinSocket套接字 1.3 WSAtartup函數 1.4 socket函數 1.5 字節序轉換 1.6 綁定套接字 1.7 監聽 1.8 連接 1.9 接收數據 1.10 發送數據 1.11 關閉套接字 二、UDP連接流程…

QT 圖表(拆線圖,欄狀圖,餅狀圖 ,動態圖表)

效果 折線圖 // 創建折線數據系列// 創建折線系列QLineSeries *series new QLineSeries;// series->append(0, 6);// series->append(2, 4);// series->append(3, 8);// 創建圖表并添加系列QChart *chart new QChart;chart->addSeries(series);chart->setTit…

vector和list的區別是什么

vector 和 list 都是C 標準模板庫&#xff08;STL&#xff09;中的容器&#xff0c;它們的區別如下&#xff1a; 內存結構 - vector &#xff1a;是連續的內存空間&#xff0c;就像數組一樣&#xff0c;元素在內存中依次排列。 - list &#xff1a;是由節點組成的雙向鏈表…

【AI】【AIGC】降低AIGC檢測率:技術、挑戰與應對策略

引言 隨著生成式人工智能&#xff08;AIGC&#xff09;技術的迅速發展&#xff0c;越來越多的內容開始由人工智能生成。AIGC技術的應用非常廣泛&#xff0c;包括文本生成、圖像生成、音頻生成等。然而&#xff0c;隨著這些技術的普及&#xff0c;如何有效識別并檢測AIGC生成的…

vue3 ts 請求封裝后端接口

一 首頁-廣告區域-小程序 首頁-廣告區域-小程序 GET/home/banner1.1 請求封裝 首頁-廣告區域 home.ts export const getHomeBannerApi (distributionSite 1) > {return http<BannerItem[]>({method: GET,url: /home/banner,data: {distributionSite,},}) }函數定…

響應式CMS架構優化SEO與用戶體驗

內容概要 在數字化內容生態中&#xff0c;響應式CMS架構已成為平衡搜索引擎可見性與終端用戶體驗的核心載體。該系統通過多終端適配技術&#xff0c;確保PC、移動端及平板等設備的內容渲染一致性&#xff0c;直接降低頁面跳出率并延長用戶停留時長。與此同時&#xff0c;智能S…

算法基礎篇(1)(藍橋杯常考點)

算法基礎篇 前言 算法內容還有搜索&#xff0c;數據結構&#xff08;進階&#xff09;&#xff0c;動態規劃和圖論 數學那個的話大家也知道比較難&#xff0c;放在最后講 這期包含的內容可以看目錄 模擬那個算法的話就是題說什么寫什么&#xff0c;就不再分入目錄中了 注意事…

MyBatis一級緩存和二級緩存

介紹 在開發基于 MyBatis 的應用時&#xff0c;緩存是提升性能的關鍵因素之一。MyBatis 提供了一級緩存和二級緩存&#xff0c;合理使用它們可以顯著減少數據庫的訪問次數&#xff0c;提高系統的響應速度和吞吐量。本文將深入探討 MyBatis 一級緩存和二級緩存的工作原理、使用…