欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

SpringBoot ThreadLocal 簡(jiǎn)單介紹及使用詳解

 更新時(shí)間:2024年01月19日 14:32:47   作者:致最長(zhǎng)的電影  
ThreadLocal 叫做線程變量,意思是 ThreadLocal 中填充的變量屬于當(dāng)前線程,該變量對(duì)其他線程而言是隔離的,也就是說該變量是當(dāng)前線程獨(dú)有的變量,這篇文章主要介紹了SpringBoot ThreadLocal 的詳解,需要的朋友可以參考下

一、ThreadLocal 簡(jiǎn)介

ThreadLocal 叫做線程變量,意思是 ThreadLocal 中填充的變量屬于當(dāng)前線程,該變量對(duì)其他線程而言是隔離的,也就是說該變量是當(dāng)前線程獨(dú)有的變量。ThreadLocal 為變量在每個(gè)線程中都創(chuàng)建了一個(gè)副本,那么每個(gè)線程可以訪問自己內(nèi)部的副本變量。

ThreadLocal 變量,線程局部變量,同一個(gè) ThreadLocal 所包含的對(duì)象,在不同的 Thread 中有不同的副本。這里有幾點(diǎn)需要注意:

  • 因?yàn)槊總€(gè) Thread 內(nèi)有自己的實(shí)例副本,且該副本只能由當(dāng)前 Thread 使用。這也是 ThreadLocal 命名的由來。
  • 既然每個(gè) Thread 有自己的實(shí)例副本,且其它 Thread 不可訪問,那就不存在多線程間共享的問題。

ThreadLocal 提供了線程本地的實(shí)例。它與普通變量的區(qū)別在于,每個(gè)使用該變量的線程都會(huì)初始化一個(gè)完全獨(dú)立的實(shí)例副本。

ThreadLocal 變量通常被 private static 修飾。當(dāng)一個(gè)線程結(jié)束時(shí),它所使用的所有 ThreadLocal 相對(duì)的實(shí)例副本都可被回收。

下圖可以增強(qiáng)理解:

 二、ThreadLocal 與 Synchronized 的區(qū)別

ThreadLocal<T> 其實(shí)是與線程綁定的一個(gè)變量。ThreadLocal 和 Synchorized 都用于解決多線程并發(fā)訪問。

但是 ThreadLocal 和 Synchorized 有本質(zhì)的區(qū)別:

  • Synchorized 用于線程間的數(shù)據(jù)共享,而 ThreadLocal 則用于線程間的數(shù)據(jù)隔離。
  • Synchorized 是利用鎖的機(jī)制,使變量或代碼塊在某一時(shí)刻只能被一個(gè)線程訪問。而 ThreadLocal 為每一個(gè)線程都提供了變量的副本,使得每個(gè)線程在某一時(shí)間訪問到的并不是同一個(gè)對(duì)象,這樣就隔離了多個(gè)線程對(duì)數(shù)據(jù)的數(shù)據(jù)共享。

而 Synchorized 卻正好相反,它用于在多個(gè)線程間通信時(shí)能夠獲得數(shù)據(jù)共享。

總結(jié):

一句話理解 ThreadLocal ,threadLocal 是作為當(dāng)前線程中屬性 ThreadLocalMap 集合中的某一個(gè) Entry 的 key 值,Entry(threadlocal,value),雖然不同的線程之間 threadLocal 這個(gè) key 值是一樣,但是不同的線程所擁有的 ThreadLocalMap 是獨(dú)一無二的,也就是不同的線程間同一個(gè) ThreadLocal(key)對(duì)應(yīng)存儲(chǔ)的值(value)不一樣,從而到達(dá)了線程間變量隔離的目的,但是在同一個(gè)線程中這個(gè) value 變量地址是一樣的。

三、ThreadLocal 的簡(jiǎn)單使用

public class ThreadLocalTest {
    private static ThreadLocal<String> localVar = new ThreadLocal<String>();
    static void print(String str) {
        // 打印當(dāng)前線程中本地內(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

從這個(gè)示例中我們可以看到,兩個(gè)線程分別獲取了自己線程存放的變量,他們之間變量的獲取并不會(huì)錯(cuò)亂。這個(gè)的理解也可以結(jié)合圖,相信會(huì)有一個(gè)更深刻地理解。

四、ThreadLocal 常見使用場(chǎng)景

ThreadLocal 適用于如下兩種場(chǎng)景

  • 每個(gè)線程需要有自己?jiǎn)为?dú)的實(shí)例
  • 實(shí)例需要在多個(gè)方法中共享,但不希望被多線程共享

對(duì)于第一點(diǎn),每個(gè)線程擁有自己實(shí)例,實(shí)現(xiàn)它的方式很多。例如可以在線程內(nèi)部構(gòu)建一個(gè)單獨(dú)的實(shí)例。ThreadLoca 可以以非常方便的形式滿足該需求。

對(duì)于第二點(diǎn),可以在滿足第一點(diǎn)(每個(gè)線程有自己的實(shí)例)的條件下,通過方法間引用傳遞的形式實(shí)現(xiàn)。ThreadLocal 使得代碼耦合度更低,且實(shí)現(xiàn)更優(yōu)雅。

存儲(chǔ)用戶 userInfo 場(chǎng)景:

@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) {
        // 校驗(yàn)代碼
        log.info("驗(yàn)證成功,保存到threadLocal userInfo={}", userInfo);
        OnlineUserUtil.set(userInfo);
    }
    @AfterReturning(value = "@annotation(tokenAuthentication)")
    public void doAfter(TokenAuthentication tokenAuthentication) {
        OnlineUserUtil.remove();
    }
}

這樣在方法中,都能用到 userInfo 這個(gè)對(duì)象。

五、如何正確的使用 ThreadLocal

  • 將ThreadLocal變量定義成private static的,這樣的話ThreadLocal的生命周期就更長(zhǎng),由于一直存在ThreadLocal的強(qiáng)引用,所以ThreadLocal也就不會(huì)被回收,也就能保證任何時(shí)候都能根據(jù)ThreadLocal的弱引用訪問到Entry的value值,然后remove它,防止內(nèi)存泄露。
  • 每次使用完ThreadLocal,都調(diào)用它的remove()方法,清除數(shù)據(jù)。

六、參考文檔

到此這篇關(guān)于SpringBoot ThreadLocal 的詳解的文章就介紹到這了,更多相關(guān)SpringBoot ThreadLocal內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 通過Java實(shí)現(xiàn)zip文件與rar文件解壓縮的詳細(xì)步驟

    通過Java實(shí)現(xiàn)zip文件與rar文件解壓縮的詳細(xì)步驟

    這篇文章主要給大家介紹了如何通過?Java?來完成?zip?文件與?rar?文件的解壓縮,文中通過代碼示例講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下
    2024-07-07
  • 前端RSA加密java后端解密示例代碼

    前端RSA加密java后端解密示例代碼

    這篇文章主要介紹了RSA非對(duì)稱加密的原理,前端使用公鑰加密數(shù)據(jù),后端使用私鑰解密,提供了前端和后端實(shí)現(xiàn)的示例代碼,包括依賴、接口、工具類等,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2024-11-11
  • JDK1.8中ConcurrentHashMap中computeIfAbsent死循環(huán)bug問題

    JDK1.8中ConcurrentHashMap中computeIfAbsent死循環(huán)bug問題

    這篇文章主要介紹了JDK1.8中ConcurrentHashMap中computeIfAbsent死循環(huán)bug,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-08-08
  • Java學(xué)習(xí)之如何進(jìn)行JSON解析

    Java學(xué)習(xí)之如何進(jìn)行JSON解析

    JSON(JavaScript?Object?Notation)是一種輕量級(jí)的數(shù)據(jù)交換格式,它算是JavaScript語言的一部分,與XML一樣都可以用于數(shù)據(jù)的存儲(chǔ)和傳輸,本文講給大家介紹如何進(jìn)行JSON解析,需要的朋友可以參考下
    2023-12-12
  • Spring處理@Async導(dǎo)致的循環(huán)依賴失敗問題的方案詳解

    Spring處理@Async導(dǎo)致的循環(huán)依賴失敗問題的方案詳解

    這篇文章主要為大家詳細(xì)介紹了SpringBoot中的@Async導(dǎo)致循環(huán)依賴失敗的原因及其解決方案,文中的示例代碼講解詳細(xì),感興趣的可以學(xué)習(xí)一下
    2022-07-07
  • mybatis中foreach報(bào)錯(cuò):_frch_item_0 not found的解決方法

    mybatis中foreach報(bào)錯(cuò):_frch_item_0 not found的解決方法

    這篇文章主要給大家介紹了mybatis中foreach報(bào)錯(cuò):_frch_item_0 not found的解決方法,文章通過示例代碼介紹了詳細(xì)的解決方法,對(duì)大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起看看吧。
    2017-06-06
  • springboot 錯(cuò)誤處理小結(jié)

    springboot 錯(cuò)誤處理小結(jié)

    在 java web開發(fā)過程中,難免會(huì)有一些系統(tǒng)異?;蛉藶楫a(chǎn)生一些異常。在 RESTful springboot 項(xiàng)目中如何優(yōu)雅的處理?下面腳本之家小編給大家?guī)砹藄pringboot 錯(cuò)誤處理小結(jié),感興趣的朋友一起看看吧
    2018-03-03
  • java 抽象類和接口的區(qū)別詳細(xì)解析

    java 抽象類和接口的區(qū)別詳細(xì)解析

    abstractclass和interface是Java語言中對(duì)于抽象類定義進(jìn)行支持的兩種機(jī)制,正是由于這兩種機(jī)制的存在,才賦予了Java強(qiáng)大的面向?qū)ο竽芰?需要了解的朋友可以參考下
    2012-11-11
  • java如何根據(jù)HttpServletRequest獲取IP地址

    java如何根據(jù)HttpServletRequest獲取IP地址

    文章介紹了幾種代理服務(wù)器轉(zhuǎn)發(fā)服務(wù)請(qǐng)求頭的方法,這些請(qǐng)求頭可能包含真實(shí)IP地址,但并不是所有的代理都會(huì)包括這些請(qǐng)求頭,而且這些IP地址可能被偽造
    2025-03-03
  • Java11?中基于嵌套關(guān)系的訪問控制優(yōu)化問題

    Java11?中基于嵌套關(guān)系的訪問控制優(yōu)化問題

    在?Java?語言中,類和接口可以相互嵌套,這種組合之間可以不受限制的彼此訪問,包括訪問彼此的構(gòu)造函數(shù)、字段、方法,接下來通過本文給大家介紹Java11中基于嵌套關(guān)系的訪問控制優(yōu)化問題,感興趣的朋友一起看看吧
    2022-01-01

最新評(píng)論