Java ThreadLocal 的使用與源碼解析

来源:https://www.cnblogs.com/cloudfloating/archive/2019/10/22/11723093.html
-Advertisement-
Play Games

GitHub Page: "http://blog.cloudli.top/posts/Java ThreadLocal 的使用與源碼解析/" 主要解決的是每個線程綁定自己的值,可以將 看成全局存放數據的盒子,盒子中存儲每個線程的私有數據。 驗證線程變數的隔離性 get() 方法 方法首先得到當前線 ...


GitHub Page: http://blog.cloudli.top/posts/Java-ThreadLocal-的使用與源碼解析/

ThreadLocal 主要解決的是每個線程綁定自己的值,可以將 ThreadLocal 看成全局存放數據的盒子,盒子中存儲每個線程的私有數據。

驗證線程變數的隔離性

import static java.lang.System.out;

public class Run {

    private static ThreadLocal<Integer> threadLocal = new ThreadLocal<>();

    static class Work extends Thread {

        @Override
        public void run() {
            threadLocal.set(0);
            for (int i = 1; i <= 5; i++) {
                // 獲取數據
                int sum = threadLocal.get();
                out.printf("%s's sum = %s\n", getName(), threadLocal.get());
                sum += i;
                // 寫回數據
                threadLocal.set(sum);
            }
            out.printf("END %s's sum = %d\n\n", getName(), threadLocal.get());
        }
    }

    public static void main(String[] args) {
        Work work1 = new Work(),
                work2 = new Work();

        work1.start();
        work2.start();
    }
}

運行結果:

Thread-0's sum = null
Thread-1's sum = null
Thread-1's sum = 1
Thread-1's sum = 3
Thread-1's sum = 6
Thread-1's sum = 10
END Thread-1's sum = 15

Thread-0's sum = 1
Thread-0's sum = 3
Thread-0's sum = 6
Thread-0's sum = 10
END Thread-0's sum = 15


Process finished with exit code 0

從結果來看,兩個線程的計算結果一致,ThreadLocal 中隔離了兩個線程的數據。

ThreadLocal 源碼解析

ThreadLocalMap 內部類

ThreadLocal 中有一個 ThreadLocalMap 內部類,所以 ThreadLocal 實際上是使用一個哈希表來存儲每個線程的數據的。

ThreadLocalMapHashMap 不同,其中 Entry 是一個弱引用,這意味著每次垃圾回收運行時都會將儲存的數據回收掉。而且它只使用了數組來存儲鍵值對。

ThreadLocalMap 中的 Entry

static class Entry extends WeakReference<ThreadLocal<?>> {
            /** The value associated with this ThreadLocal. */
            Object value;

            Entry(ThreadLocal<?> k, Object v) {
                super(k);
                value = v;
            }
        }

get() 方法

public T get() {
    // 得到當前線程
    Thread t = Thread.currentThread();
    // 獲取當前線程的哈希表
    ThreadLocalMap map = getMap(t);
    if (map != null) {
        // 從哈希表中獲取當前線程的數據
        ThreadLocalMap.Entry e = map.getEntry(this);
        if (e != null) {
            @SuppressWarnings("unchecked")
            T result = (T)e.value;
            return result;
        }
    }
    return setInitialValue();
}

get() 方法首先得到當前線程,然後獲取當前線程的 ThreadLocalMap 對象,然後從中取出數據。

這裡的 map.getEntry(this) 看起來很奇怪,在前面有這樣一行代碼:

ThreadLocalMap map = getMap(t);

這個方法獲取當前線程的 ThreadLocalMap 對象,所以,雖然 map.getEntry() 中的 key 總是 ThreadLocal 對象本身,但是每個線程都持有有自己的 ThreadLocalMap 對象。

getMap() 方法

/**
 * Get the map associated with a ThreadLocal. Overridden in
 * InheritableThreadLocal.
 *
 * @param  t the current thread
 * @return the map
 */
ThreadLocalMap getMap(Thread t) {
    return t.threadLocals;
}

看到這個方法,get() 方法中 map.getEntry(this) 的迷霧就解開了。這裡可以看到返回的是線程中的 threadLocals 屬性。那麼這裡瞟一眼 Thread 的源碼:

/* ThreadLocal values pertaining to this thread. This map is maintained
 * by the ThreadLocal class. */
ThreadLocal.ThreadLocalMap threadLocals = null;

ThreadLocal

其實每次 get() 時都是先獲取了線程自己的 ThreadLocalMap 對象,然後對這個對象進行操作。

set() 方法

public void set(T value) {
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    if (map != null)
        map.set(this, value);
    else
        // 為當前線程創建一個 ThreadLocalMap 對象
        createMap(t, value);
}

set() 方法也是先獲取當前線程自己的 ThreadLocalMap 對象,然後再設置數據。如果獲取的哈希表為 null,則創建一個。

createMap() 方法

void createMap(Thread t, T firstValue) {
    t.threadLocals = new ThreadLocalMap(this, firstValue);
}

createMap() 方法創建一個 ThreadLocalMap 對象,該對象由線程持有。

總結

  • ThreadLocal 可以隔離線程的變數,每個線程只能從這個對象中獲取到屬於自己的數據。
  • ThreadLocal 使用哈希表來存儲線程的數據,而且這個哈希表是由線程自己持有的,每次獲取和設值都會先獲取當前線程持有的ThreadLocalMap 對象。
  • ThreadLocalMap 中的 key 總是 ThreadLocal 對象本身。
  • ThreadLocalMap 中的 Entry 是弱引用,每次 GC 運行都會被回收。

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

-Advertisement-
Play Games
更多相關文章
  • 一、安裝Python3.5以上版本。 鏈接:https://www.python.org/downloads/windows/ 二、安裝PyQt5。 使用pip安裝:1、進入 C:\Users\你的電腦用戶名。 2、新建一個文件夾,命名為pip。 3、進入文件夾新建一個文本文件,命名pip,文件後 ...
  • 作為開發工程師,你也許會在日常編程中被 Java 的啟動性能和記憶體耗費所震驚,繼而對 Java 語言產生懷疑;或許在使用虛擬機遇見記憶體溢出等一系列異常時頭疼萬分,困擾於為什麼會出現各種問題。 和語言朝夕相處的開發者們,提及代碼的詳細運行過程也難免會一時語塞。這都是由於 Java 虛擬機封裝得太好,讓 ...
  • 近幾年來,python的包管理系統pip 越來越完善, 尤其是對於 windows場景下,pip大大改善了python的易用性。 https://www.cnblogs.com/yvivid/p/pip_setup.html 一、python 包管理工具 pip 安裝 從 python 3.4開始就 ...
  • Object類 toString()方法 將一個對象返回為字元串形式 equals()方法 基本數據類型 比較的值 引用數據類型 比較地址值 Obejects類的equals()方法 兩對象比較,可以防止空指針異常 Date類 表示特定瞬間,精確到毫秒 把日期轉換為毫秒 當前日期 日期原點 1970 ...
  • 如果我們在spring框架中配置了多個xml文件,我們可以在讀取配置文件的時候把這些xml文件一下全都讀取 也可以只讀一個總的xml文件,在這個總的xml文件中把其他的xml全都都導入進來。 例如: student.xml文件: teacher.xml文件: import.xml文件: main: ...
  • 增強 SpringBoot 快速開發工具 項目地址: "https://gitee.com/sanri/web ui" 優點:這是一個 web 通用配置的組件,即插即用,可用於新項目或私活。是對 SpringBoot 快速開發的一種補充,它內置了大量的配置來簡化開發,遵循約定高於配置原則。 它解決的 ...
  • 線程封閉 在多線程的環境中,我們經常使用鎖來保證線程的安全,但是對於每個線程都要用的資源使用鎖的話那麼程式執行的效率就會受到影響,這個時候可以把這些資源變成線程封閉的形式。 1、棧封閉 所謂的棧封閉其實就是使用局部變數存放資源,我們知道局部變數在記憶體中是存放在虛擬機棧中,而棧又是每個線程私有獨立的, ...
  • 1.題目: 企業發放的獎金根據利潤提成。利潤(I)低於或等於10萬元時,獎金可提10%; 利潤高於10萬元,低於20萬元時,低於10萬元的部分按10%提成,高於10萬元的部分,可提成7.5%; 20萬到40萬之間時,高於20萬元的部分,可提成5%;40萬到60萬之間時高於40萬元的部分,可提成3%; ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...