SpringBoot ThreadLocal 簡單介紹及使用詳解
一、ThreadLocal 簡介
ThreadLocal 叫做線程變量,意思是 ThreadLocal 中填充的變量屬于當前線程,該變量對其他線程而言是隔離的,也就是說該變量是當前線程獨有的變量。ThreadLocal 為變量在每個線程中都創(chuàng)建了一個副本,那么每個線程可以訪問自己內(nèi)部的副本變量。
ThreadLocal 變量,線程局部變量,同一個 ThreadLocal 所包含的對象,在不同的 Thread 中有不同的副本。這里有幾點需要注意:
- 因為每個 Thread 內(nèi)有自己的實例副本,且該副本只能由當前 Thread 使用。這也是 ThreadLocal 命名的由來。
- 既然每個 Thread 有自己的實例副本,且其它 Thread 不可訪問,那就不存在多線程間共享的問題。
ThreadLocal 提供了線程本地的實例。它與普通變量的區(qū)別在于,每個使用該變量的線程都會初始化一個完全獨立的實例副本。
ThreadLocal 變量通常被 private static 修飾。當一個線程結(jié)束時,它所使用的所有 ThreadLocal 相對的實例副本都可被回收。
下圖可以增強理解:
二、ThreadLocal 與 Synchronized 的區(qū)別
ThreadLocal<T> 其實是與線程綁定的一個變量。ThreadLocal 和 Synchorized 都用于解決多線程并發(fā)訪問。
但是 ThreadLocal 和 Synchorized 有本質(zhì)的區(qū)別:
- Synchorized 用于線程間的數(shù)據(jù)共享,而 ThreadLocal 則用于線程間的數(shù)據(jù)隔離。
- Synchorized 是利用鎖的機制,使變量或代碼塊在某一時刻只能被一個線程訪問。而 ThreadLocal 為每一個線程都提供了變量的副本,使得每個線程在某一時間訪問到的并不是同一個對象,這樣就隔離了多個線程對數(shù)據(jù)的數(shù)據(jù)共享。
而 Synchorized 卻正好相反,它用于在多個線程間通信時能夠獲得數(shù)據(jù)共享。
總結(jié):
一句話理解 ThreadLocal ,threadLocal 是作為當前線程中屬性 ThreadLocalMap 集合中的某一個 Entry 的 key 值,Entry(threadlocal,value),雖然不同的線程之間 threadLocal 這個 key 值是一樣,但是不同的線程所擁有的 ThreadLocalMap 是獨一無二的,也就是不同的線程間同一個 ThreadLocal(key)對應存儲的值(value)不一樣,從而到達了線程間變量隔離的目的,但是在同一個線程中這個 value 變量地址是一樣的。
三、ThreadLocal 的簡單使用
public class ThreadLocalTest { private static ThreadLocal<String> localVar = new ThreadLocal<String>(); static void print(String str) { // 打印當前線程中本地內(nèi)存中變量的值 System.out.println(str + ":" + localVar.get()); // 清除內(nèi)存中的本地變量 localVar.remove(); } public static void main(String[] args) throws InterruptedException { new Thread(new Runnable() { @Override public void run() { ThreadLocalTest.localVar.set("xdclass_A"); print("A"); // 打印本地變量 System.out.println("清楚后:" + localVar.get()); } }, "A").start(); Thread.sleep(1000); new Thread(new Runnable() { @Override public void run() { ThreadLocalTest.localVar.set("xdclass_B"); print("B"); // 打印本地變量 System.out.println("清楚后:" + localVar.get()); } }, "B").start(); } } A:xdclass_A 清楚后:null B:xdclass_B 清楚后:null
從這個示例中我們可以看到,兩個線程分別獲取了自己線程存放的變量,他們之間變量的獲取并不會錯亂。這個的理解也可以結(jié)合圖,相信會有一個更深刻地理解。
四、ThreadLocal 常見使用場景
ThreadLocal 適用于如下兩種場景
- 每個線程需要有自己單獨的實例
- 實例需要在多個方法中共享,但不希望被多線程共享
對于第一點,每個線程擁有自己實例,實現(xiàn)它的方式很多。例如可以在線程內(nèi)部構(gòu)建一個單獨的實例。ThreadLoca 可以以非常方便的形式滿足該需求。
對于第二點,可以在滿足第一點(每個線程有自己的實例)的條件下,通過方法間引用傳遞的形式實現(xiàn)。ThreadLocal 使得代碼耦合度更低,且實現(xiàn)更優(yōu)雅。
存儲用戶 userInfo 場景:
@Slf4j public class OnlineUserUtil { private final static ThreadLocal<UserInfo> threadLocal = new ThreadLocal<>(); public static void set(UserInfo userInfo) { threadLocal.set(userInfo); } public static UserInfo get() { return threadLocal.get(); } public static void remove() { threadLocal.remove(); } }
@Slf4j @Aspect @Component @Order(2) public class TokenAuthenticationAspect { @Before(value = "@annotation(tokenAuthentication)") public void doBefore(JoinPoint pjp, TokenAuthentication tokenAuthentication) { // 校驗代碼 log.info("驗證成功,保存到threadLocal userInfo={}", userInfo); OnlineUserUtil.set(userInfo); } @AfterReturning(value = "@annotation(tokenAuthentication)") public void doAfter(TokenAuthentication tokenAuthentication) { OnlineUserUtil.remove(); } }
這樣在方法中,都能用到 userInfo 這個對象。
五、如何正確的使用 ThreadLocal
- 將ThreadLocal變量定義成private static的,這樣的話ThreadLocal的生命周期就更長,由于一直存在ThreadLocal的強引用,所以ThreadLocal也就不會被回收,也就能保證任何時候都能根據(jù)ThreadLocal的弱引用訪問到Entry的value值,然后remove它,防止內(nèi)存泄露。
- 每次使用完ThreadLocal,都調(diào)用它的remove()方法,清除數(shù)據(jù)。
六、參考文檔
到此這篇關(guān)于SpringBoot ThreadLocal 的詳解的文章就介紹到這了,更多相關(guān)SpringBoot ThreadLocal內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- springboot在filter中如何用threadlocal存放用戶身份信息
- SpringBoot中的ThreadLocal保存請求用戶信息的實例demo
- springboot登錄攔截器+ThreadLocal實現(xiàn)用戶信息存儲的實例代碼
- SpringBoot+ThreadLocal+AbstractRoutingDataSource實現(xiàn)動態(tài)切換數(shù)據(jù)源
- Springboot公共字段填充及ThreadLocal模塊改進方案
- SpringBoot ThreadLocal實現(xiàn)公共字段自動填充案例講解
- SpringBoot通過ThreadLocal實現(xiàn)登錄攔截詳解流程
- springboot 使用ThreadLocal的實例代碼
- SpringBoot中使用?ThreadLocal?進行多線程上下文管理及注意事項小結(jié)
相關(guān)文章
通過Java實現(xiàn)zip文件與rar文件解壓縮的詳細步驟
這篇文章主要給大家介紹了如何通過?Java?來完成?zip?文件與?rar?文件的解壓縮,文中通過代碼示例講解的非常詳細,對大家的學習或工作有一定的幫助,需要的朋友可以參考下2024-07-07JDK1.8中ConcurrentHashMap中computeIfAbsent死循環(huán)bug問題
這篇文章主要介紹了JDK1.8中ConcurrentHashMap中computeIfAbsent死循環(huán)bug,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-08-08Spring處理@Async導致的循環(huán)依賴失敗問題的方案詳解
這篇文章主要為大家詳細介紹了SpringBoot中的@Async導致循環(huán)依賴失敗的原因及其解決方案,文中的示例代碼講解詳細,感興趣的可以學習一下2022-07-07mybatis中foreach報錯:_frch_item_0 not found的解決方法
這篇文章主要給大家介紹了mybatis中foreach報錯:_frch_item_0 not found的解決方法,文章通過示例代碼介紹了詳細的解決方法,對大家具有一定的參考學習價值,需要的朋友們下面來一起看看吧。2017-06-06java如何根據(jù)HttpServletRequest獲取IP地址
文章介紹了幾種代理服務器轉(zhuǎn)發(fā)服務請求頭的方法,這些請求頭可能包含真實IP地址,但并不是所有的代理都會包括這些請求頭,而且這些IP地址可能被偽造2025-03-03Java11?中基于嵌套關(guān)系的訪問控制優(yōu)化問題
在?Java?語言中,類和接口可以相互嵌套,這種組合之間可以不受限制的彼此訪問,包括訪問彼此的構(gòu)造函數(shù)、字段、方法,接下來通過本文給大家介紹Java11中基于嵌套關(guān)系的訪問控制優(yōu)化問題,感興趣的朋友一起看看吧2022-01-01