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

Java線程中的ThreadLocal類解讀

 更新時(shí)間:2023年11月28日 10:30:22   作者:后仰大風(fēng)車  
這篇文章主要介紹了Java線程中的ThreadLocal類解讀,ThreadLocal是一個(gè)泛型類,作用是實(shí)現(xiàn)線程隔離,ThreadLocal類型的變量,在每個(gè)線程中都會(huì)對(duì)應(yīng)一個(gè)具體對(duì)象,對(duì)象類型需要在聲明ThreadLocal變量時(shí)指定,需要的朋友可以參考下

Java的ThreadLocal類

ThreadLocal是一個(gè)泛型類,作用是實(shí)現(xiàn)線程隔離,ThreadLocal類型的變量,在每個(gè)線程中都會(huì)對(duì)應(yīng)一個(gè)具體對(duì)象,對(duì)象類型需要在聲明ThreadLocal變量時(shí)指定。

ThreadLocal的使用示例

package org.example.thread;

import org.example.domain.Book;

public class ThreadLocalTest {

    ThreadLocal<Book> localBook = new ThreadLocal<>();

    Book book = new Book();

    public void test() {

        Thread thread = new Thread(new Runnable() {
            @Override
            public void run() {
                localBook.set(book);
                System.out.println(localBook.get());
                localBook.remove();
            }
        });
        thread.start();

        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
        System.out.println(localBook.get());

        localBook.remove();
    }

    public static void main(String[] arg) {
        ThreadLocalTest localBookTest = new ThreadLocalTest();
        localBookTest.test();
    }
}

運(yùn)行結(jié)果如下:

在這里插入圖片描述

示例代碼中,只聲明了一個(gè) ThreadLocal 變量localBook,在子線程中設(shè)置了localBook的值,但是在最后主線程中進(jìn)行打印時(shí),發(fā)現(xiàn)為null,和子線程中的結(jié)果不一樣。

具體原因還要看看ThreadLocal類,以及內(nèi)部 set、get等方法的實(shí)現(xiàn)。

原理

1、ThreadLocal中有一個(gè)靜態(tài)內(nèi)部類 ThreadLocalMap,ThreadLocalMap中維護(hù)了一個(gè) Entry數(shù)組,Entry又是ThreadLocalMap的內(nèi)部類,用來表示一個(gè)KV鍵值對(duì)。部分源碼如下:

    static class ThreadLocalMap {

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

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

        private static final int INITIAL_CAPACITY = 16;

        private Entry[] table;

Entry的 key是一個(gè)指向ThreadLocal對(duì)象的弱引用,value則指向與key對(duì)應(yīng)的Object對(duì)象。

2、Thread類中,有一個(gè)ThreadLocalMap類型的成員變量,初始為null:

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

也就是說,每個(gè)線程對(duì)象中,都有一個(gè)ThreadLocalMap,它里面可以有很多個(gè)Entry,每個(gè)Entry中的key都指向一個(gè)ThreadLocal對(duì)象,value則指向與key相對(duì)應(yīng)的Object對(duì)象。在線程中調(diào)用ThreadLocal對(duì)象的set、get方法,實(shí)際操作的是當(dāng)前線程自己的ThreadLocalMap。

3、set方法

    public void set(T value) {
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null) {
            map.set(this, value);
        } else {
            createMap(t, value);
        }
    }

    ThreadLocalMap getMap(Thread t) {
        return t.threadLocals;
    }

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

果然,這里先拿到當(dāng)前線程中的ThreadLocalMap,如果不為空則直接添加一組鍵值對(duì),key是當(dāng)前的ThreadLocal變量; 如果map為空,則新建一個(gè)map,同樣添加一組鍵值對(duì),key是當(dāng)前的ThreadLocal變量,然后將map賦值給當(dāng)前線程的ThreadLocalMap。

4、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中,與當(dāng)前ThreadLocal變量對(duì)應(yīng)的value對(duì)象。

5、remove方法

     public void remove() {
         ThreadLocalMap m = getMap(Thread.currentThread());
         if (m != null) {
             m.remove(this);
         }
     }

        private void remove(ThreadLocal<?> key) {
            Entry[] tab = table;
            int len = tab.length;
            int i = key.threadLocalHashCode & (len-1);
            for (Entry e = tab[i];
                 e != null;
                 e = tab[i = nextIndex(i, len)]) {
                if (e.get() == key) {
                    e.clear();
                    expungeStaleEntry(i);
                    return;
                }
            }
        }

remove方法,移除當(dāng)前線程的ThreadLocalMap中,與當(dāng)前ThreadLocal對(duì)象對(duì)應(yīng)的鍵值對(duì)(key、value及Entry置空)。

內(nèi)存泄漏相關(guān)

1、為什么ThreadLocalMap中的key要使用弱引用?

弱引用特性:在對(duì)象沒有被強(qiáng)引用指向,而僅被弱引用指向的情況下,發(fā)生垃圾回收時(shí),會(huì)直接清理掉該對(duì)象。 套用網(wǎng)上出現(xiàn)較多的一張圖片來表示ThreadLocal變量的內(nèi)存結(jié)構(gòu):

在這里插入圖片描述

這里實(shí)線表示強(qiáng)引用,虛線表示弱引用。 假設(shè)key也是強(qiáng)引用,如果我們把 ThreadLocal Ref這個(gè)引用置為null,表示我們不再需要這個(gè)ThreadLocal對(duì)象,那么發(fā)生垃圾回收時(shí),這個(gè)變量理應(yīng)被回收掉;但實(shí)際并非如此,因?yàn)楫?dāng)前線程中還持有一個(gè)強(qiáng)引用,也就是ThreadLocalMap中的key還在指向它,那么只要線程不結(jié)束,這個(gè)沒有實(shí)際作用的ThreadLocal對(duì)象就一直不會(huì)被回收,從而出現(xiàn)內(nèi)存泄漏。而將key設(shè)計(jì)為弱引用,就能保證這種情況下ThreadLocal對(duì)象被回收,一定程度避免內(nèi)存泄漏問題。

2、為什么使用完ThreadLocal對(duì)象,要在線程中調(diào)用remove方法? 雖然弱引用可以使不再使用的ThreadLocal對(duì)象被回收掉,但還有一個(gè)問題: Entry中的key被置為了null,對(duì)應(yīng)的value已經(jīng)無法通過key訪問到,然而Current thread -> ThreadLocalMap -> Entry(value) -> my value 這條強(qiáng)引用鏈仍然存在,也就是說還存在key為null,但value不為null的entry,如果這類entry不被清理掉,還是會(huì)導(dǎo)致內(nèi)存泄露。

key為null的entry如何清理? 在ThreadLocalMap的Entry數(shù)組中,key為null的entry也會(huì)占用一個(gè)數(shù)組下標(biāo),而Threadlocal的set、get等方法,正是根據(jù)key的hashCode計(jì)算得到數(shù)組下標(biāo),然后根據(jù)下標(biāo)找到對(duì)應(yīng)的entry進(jìn)行操作。 為了維護(hù)ThreadLocalMap的可用性,不讓這些key為null的無用entry占用過多空間,set、get方法在某些情況下也會(huì)對(duì)key為null的entry進(jìn)行清理。但是用戶有可能不再需要調(diào)用其他ThreadLocal變量的set或get方法,所以這種方式是被動(dòng)且沒有針對(duì)性的。

建議在不使用ThreadLocal對(duì)象之后,直接調(diào)用remove方法,其作用就是將對(duì)應(yīng)entry的key、value置空,并將entry從數(shù)組中移除,切斷下圖中的這三處引用關(guān)系,防止內(nèi)存泄漏。

在這里插入圖片描述

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

相關(guān)文章

  • ThreadLocal線程在Java框架中的應(yīng)用及原理深入理解

    ThreadLocal線程在Java框架中的應(yīng)用及原理深入理解

    這篇文章主要介紹了ThreadLocal在Java框架中的應(yīng)用及原理深入理解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2024-01-01
  • spring+netty服務(wù)器搭建的方法

    spring+netty服務(wù)器搭建的方法

    本篇文章主要介紹了spring+netty服務(wù)器搭建的方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-01-01
  • spring cloud gateway集成hystrix實(shí)戰(zhàn)篇

    spring cloud gateway集成hystrix實(shí)戰(zhàn)篇

    這篇文章主要介紹了spring cloud gateway集成hystrix實(shí)戰(zhàn),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-07-07
  • ssm 使用token校驗(yàn)登錄的實(shí)現(xiàn)

    ssm 使用token校驗(yàn)登錄的實(shí)現(xiàn)

    這篇文章主要介紹了ssm 使用token校驗(yàn)登錄的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-04-04
  • SpringBoot整合Redis的步驟

    SpringBoot整合Redis的步驟

    這篇文章主要介紹了SpringBoot整合Redis的步驟,幫助大家更好的理解和使用springboot框架,感興趣的朋友可以了解下
    2020-11-11
  • java利用phantomjs進(jìn)行截圖實(shí)例教程

    java利用phantomjs進(jìn)行截圖實(shí)例教程

    PlantomJs是一個(gè)基于javascript的webkit內(nèi)核無頭瀏覽器 也就是沒有顯示界面的瀏覽器,你可以在基于 webkit 瀏覽器做的事情,它都能做到。下面這篇文章主要給大家介紹了關(guān)于java利用phantomjs進(jìn)行截圖的相關(guān)資料,需要的朋友可以參考下
    2018-10-10
  • Java中的Set、List、Map的用法與區(qū)別介紹

    Java中的Set、List、Map的用法與區(qū)別介紹

    這篇文章主要介紹了Java中的Set、List、Map的用法與區(qū)別,需要的朋友可以參考下
    2016-06-06
  • Java實(shí)現(xiàn)的矩陣乘法示例

    Java實(shí)現(xiàn)的矩陣乘法示例

    這篇文章主要介紹了Java實(shí)現(xiàn)的矩陣乘法,簡(jiǎn)單描述了矩陣乘法的原理,并結(jié)合實(shí)例形式分析了java實(shí)現(xiàn)矩陣乘法的相關(guān)操作技巧,需要的朋友可以參考下
    2019-03-03
  • Java中JDBC的使用教程詳解

    Java中JDBC的使用教程詳解

    Java語言操作數(shù)據(jù)庫(kù)?JDBC本質(zhì):其實(shí)是官方(sun公司)定義的一套操作所有關(guān)系型數(shù)據(jù)庫(kù)的規(guī)則,即接口。本文講解了JDBC的使用方法,需要的可以參考一下
    2022-06-06
  • 使用Nacos作為配置中心的命名空間、配置分組

    使用Nacos作為配置中心的命名空間、配置分組

    文章詳細(xì)介紹了Spring Cloud Config配置中心的命名空間、配置集、配置集ID、配置分組以及如何在微服務(wù)中加載和使用這些配置,通過配置中心,可以實(shí)現(xiàn)配置隔離和集中管理,簡(jiǎn)化微服務(wù)的配置維護(hù)
    2024-12-12

最新評(píng)論