HashMap(get和put)jdk8

来源:https://www.cnblogs.com/tie-dao/archive/2022/08/31/16638355.html
-Advertisement-
Play Games

get邏輯: HashMap數據結構為數組加鏈表加紅黑樹、只有當鏈表數量大於8時、才將鏈表轉換為紅黑樹、時間複雜度由鏈表的O(N)轉換為紅黑樹的O(logN) // 主要看getNode下的方法、傳入key的hash值和key public V get(Object key) { Node<K,V> ...


get邏輯:

HashMap數據結構為數組加鏈表加紅黑樹、只有當鏈表數量大於8時、才將鏈表轉換為紅黑樹、時間複雜度由鏈表的O(N)轉換為紅黑樹的O(logN)

// 主要看getNode下的方法、傳入key的hash值和key
public V get(Object key) {
    Node<K,V> e;
    return (e = getNode(hash(key), key)) == null ? null : e.value;
}

// 返回一個Node對象、包含了key和value、在get方法中在返回value值
final Node<K,V> getNode(int hash, Object key) {
    // tab:Node<K, V>對象數組
    Node<K,V>[] tab;
    // first: 指向key hash值對應的數組值 e: first對應Node對象的下一個節點
    Node<K,V> first, e; 
    // n: 指向當前HashMpa的數組長度
    int n;
    // k: 臨時變數、指向 key
    K k;
    // 這裡檢測HashMap對象的數組是否存在、長度是否大於0、((n - 1) & hash)根據此表達式算出key對應的數組位置、在檢查是否存在對象。
    if (
        (tab = table) != null && 
        (n = tab.length) > 0 &&
        (first = tab[(n - 1) & hash]) != null
    ) {
        // first當前值為一個Node節點
        // 這個if是檢測當前的first指向的Node是否是要獲取的對象
        // 直接判斷first的hash值和要獲取的hash值是否一直、並且key的值是否一直、通過 ==判斷地址!=null和equals判斷、key賦值為first的key
        if (first.hash == hash && ((k = first.key) == key || (key != null && key.equals(k))))
            // 判斷一致後直接返回要獲取的Node節點給get、get在返回Node的value值
            return first;
        // 由於上面查找都查找不到、所以要查找Node的下一個節點、即查詢鏈表或者紅黑樹
        if ((e = first.next) != null) {
            // 檢查first對象是否是TreeNode(紅黑樹)
            if (first instanceof TreeNode)
                // 當前first為紅黑樹對象、直接根據key調用內部的檢索方法獲取對應的value
                return ((TreeNode<K,V>)first).getTreeNode(hash, key);
            do {
                // 鏈表查詢、由於first上面判斷過不是要查找的對象、e在上面語句已經指向first下一個節點、所以直接開始判斷
                // 和上面的判斷一樣、檢查hash值和key、qeuals判斷、如果有則返回對應的Node對象、沒有則最終執行下麵的return null;語句。
                if (e.hash == hash &&
                    ((k = e.key) == key || (key != null && key.equals(k))))
                    return e;
            } while ((e = e.next) != null);
        }
    }
    // 表示當前對象並沒有存儲相關的key值、返回null
    return null;
}

put邏輯

public V put(K key, V value) {
    // 內部調用putVal設置值、參數如下int hash, K key, V value, boolean onlyIfAbsent(如果為 true,則不更改現有值),boolean evict(如果為 false,則表處於創建模式)
    return putVal(hash(key), key, value, false, true);
}

// 參數如下int hash, K key, V value, boolean onlyIfAbsent(如果為 true,則不更改現有值),boolean evict(如果為 false,則表處於創建模式)
final V putVal(int hash, K key, V value, boolean onlyIfAbsent,
               boolean evict) {
    // tab指向當前HashMap對象數組
    Node<K,V>[] tab; 
    // p指向key的hash值所在的數組Node對象
    Node<K,V> p;
    // n:HashMap數組的長度、i:key的hash值對應的索引(index)
    int n, i;
    // 判斷當前HashMap的數組對象是否為空、並且長度是否為0
    if ((tab = table) == null || (n = tab.length) == 0)
        // 分配數組空間並把長度返回給n
        n = (tab = resize()).length;
    // 計算hash對應的索引是否有對象存在、沒有的話則創建Node對象、並將要put的值寫入Node對象、在返回給數組
    if ((p = tab[i = (n - 1) & hash]) == null)
        tab[i] = newNode(hash, key, value, null);
    // 要將Node對象寫入到鏈表或者紅黑樹中
    else {
        // e: 代表最終你要寫入的Node對象
        Node<K,V> e; 
        // k: 指向hash值對象的Node節點的key
        K k;
        // 檢查是否是同一個hash值、key是否相對或者進行equals判斷
        if (p.hash == hash &&
            ((k = p.key) == key || (key != null && key.equals(k))))
            // 代表同一個對象、賦值給e、最後在寫入值
            e = p;
        // 檢測是否是紅黑樹節點
        else if (p instanceof TreeNode)
            // 檢測是紅黑樹對象、直接調用內部的寫入方法、在返回一個Node<K, V>節點對象、最後在寫入值、putTreeVal裡面其實已經寫入value了、後面在寫入一次。
            e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);
        else {
            // 鏈表操作、binCount檢測有多少個鏈表節點、根據TREEIFY_THRESHOLD常量設定的值8、超過8個鏈表節點、則將該鏈表轉換為紅黑樹
            for (int binCount = 0; ; ++binCount) {
                // 
                if ((e = p.next) == null) {
                    p.next = newNode(hash, key, value, null);
                    if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st
                        // 將數組傳入
                        treeifyBin(tab, hash);
                    break;
                }
                // 檢測當前插入的對象是否一致、一致的話直接返回e、下麵在將要寫入的值賦值給變數e對象
                if (e.hash == hash &&
                    ((k = e.key) == key || (key != null && key.equals(k))))
                    break;
                p = e;
            }
        }
        // 檢測e對象是否不為空、不為空則下麵寫入對應的value值
        if (e != null) { // existing mapping for key
            V oldValue = e.value;
            // onlyIfAbsent為true表示不更新對象
            if (!onlyIfAbsent || oldValue == null)
                // 將值寫入
                e.value = value;
            afterNodeAccess(e);
            return oldValue;
        }
    }
    ++modCount;
    // 當前數組長度自增1大於上次擴容長度後、重新擴容並且把重新擴容的大小賦值給threshold
    if (++size > threshold)
        // 重新調整數組長度
        resize();
    // LinkedHashMap中重寫了、HashMap中沒有具體實現
    afterNodeInsertion(evict);
    // 對應的key無法寫入、返回null       
    return null;
}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 函數是組織代碼的非常有效的方式,有了函數,我們就可以編寫大規模的項目。可以說,函數是組織代碼的最小單元。 Python函數的定義 函數是代碼封裝的一種手段,函數中包含一段可以重覆執行的代碼,在需要用到這些代碼時,只需要調用函數,就會運行函數中的代碼。 python 函數這麼定義: def 函數名稱( ...
  • 創作不易,感謝支持! fopen函數 頭文件:stdio.h 功能是打開一個文件,其聲明格式是: FILE *fopen(const char *filename, const char *mode); 文件指針名 = fopen(文件名,使用文件方式) “文件名”是被打開文件的文件名,類型是C風格 ...
  • GUI:Graphical User Interface(圖形用戶介面) 用圖形的方式,用來顯示電腦操作的界面 Java為GUI提供的API都存在java.awt和javax.Swing兩個包中 java.awt 包: awt是這三個單詞首字母的縮寫,翻譯過來是抽象視窗工具包,只不過這個包的API ...
  • (這裡寫自定義目錄標題)Java開發入門 博客內容是本人自學java過程,所以具體工具的下載步驟會省略。其中的部分下載和安裝步驟,引用了其他博主的相關文章。 Java語言 Java是目前世界上最流行的電腦編程語言,是一種可以編寫跨平臺應用軟體的面向對象的程式設計語言,也是當今使用率最高的編程語言。 ...
  • 前兩天從網上採集到一條短視頻數據(刷短視頻),發現六公主連排5部劉亦菲主演的電影!甚是震驚,太有牌面了,看了一下日子是8月25號,嗷,原來當天是劉亦菲的生日。巧了,正好也是我家柴犬旺財的3歲生日😀。 言歸正傳,我們看到這條數據的 標題:#劉亦菲35歲生日獲央視獨寵# 神仙姐姐生日快樂! 為了分析數 ...
  • 首先,進行springboot2.7之後,官方不推薦使用/META-INF/spring.factories,轉成和SPI比較類似的/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports文件, ...
  • 近日,項目中有一個耗時較長的Job存在CPU占用過高的問題,經排查發現,主要時間消耗在往MyBatis中批量插入數據。mapper configuration是用foreach迴圈做的,差不多是這樣。(由於項目保密,以下代碼均為自己手寫的demo代碼) <insert id="batchInsert ...
  • 鏡像倉庫管理 docker倉庫,用來管理鏡像。主要分為公共倉庫和私人倉庫。下麵介紹了公共倉庫Docker Hub、私人倉庫Registry和harbor。 DockerHUb倉庫管理 什麼是DockerHUb 保存和分發鏡像的最直接方法就是使用 Docker Hub。 ​ Docker Hub 是 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...