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

淺談Java中ThreadLocal內(nèi)存泄露的原因及處理方式

 更新時(shí)間:2023年05月17日 11:27:30   作者:倔強(qiáng)的不服  
內(nèi)存泄漏就是我們申請了內(nèi)存,但是該內(nèi)存一直無法釋放,就會(huì)導(dǎo)致內(nèi)存溢出問題,本文詳細(xì)的介紹了ThreadLocal內(nèi)存泄露的原因及處理方式,感興趣的可以了解一下

1、ThreadLocal 使用原理

前文我們講過ThreadLocal的主要用途是實(shí)現(xiàn)線程間變量的隔離,表面上他們使用的是同一個(gè)ThreadLocal, 但是實(shí)際上使用的值value卻是自己獨(dú)有的一份。用一圖直接表示threadlocal 的使用方式。

圖1

從圖中我們可以當(dāng)線程使用threadlocal 時(shí),是將threadlocal當(dāng)做當(dāng)前線程thread的屬性ThreadLocalMap 中的一個(gè)Entry的key值,實(shí)際上存放的變量是Entry的value值,我們實(shí)際要使用的值是value值。value值為什么不存在并發(fā)問題呢,因?yàn)樗挥幸粋€(gè)線程能訪問。threadlocal我們可以當(dāng)做一個(gè)索引看待,可以有多個(gè)threadlocal 變量,不同的threadlocal對應(yīng)于不同的value值,他們之間互不影響。ThreadLocal為每一個(gè)線程都提供了變量的副本,使得每個(gè)線程在某一時(shí)間訪問到的并不是同一個(gè)對象,這樣就隔離了多個(gè)線程對數(shù)據(jù)的數(shù)據(jù)共享。

2、ThreadLocal 內(nèi)存泄露的原因

 Entry將ThreadLocal作為Key,值作為value保存,它繼承自WeakReference,注意構(gòu)造函數(shù)里的第一行代碼super(k),這意味著ThreadLocal對象是一個(gè)「弱引用」??梢钥磮D1.

static class Entry extends WeakReference<ThreadLocal<?>> {
    /** The value associated with this ThreadLocal. */
    Object value;
    Entry(ThreadLocal<?> k, Object v) {
        super(k);
        value = v;
    }
}

主要兩個(gè)原因
1 . 沒有手動(dòng)刪除這個(gè) Entry
2 . CurrentThread 當(dāng)前線程依然運(yùn)行

第一點(diǎn)很好理解,只要在使用完下 ThreadLocal ,調(diào)用其 remove 方法刪除對應(yīng)的 Entry ,就能避免內(nèi)存泄漏。

第二點(diǎn)稍微復(fù)雜一點(diǎn),由于ThreadLocalMap 是 Thread 的一個(gè)屬性,被當(dāng)前線程所引用,所以ThreadLocalMap的生命周期跟 Thread 一樣長。如果threadlocal變量被回收,那么當(dāng)前線程的threadlocal 變量副本指向的就是key=null, 也即entry(null,value),那這個(gè)entry對應(yīng)的value永遠(yuǎn)無法訪問到。實(shí)際私用ThreadLocal場景都是采用線程池,而線程池中的線程都是復(fù)用的,這樣就可能導(dǎo)致非常多的entry(null,value)出現(xiàn),從而導(dǎo)致內(nèi)存泄露。

綜上, ThreadLocal 內(nèi)存泄漏的根源是:

由于ThreadLocalMap 的生命周期跟 Thread 一樣長,對于重復(fù)利用的線程來說,如果沒有手動(dòng)刪除(remove()方法)對應(yīng) key 就會(huì)導(dǎo)致entry(null,value)的對象越來越多,從而導(dǎo)致內(nèi)存泄漏.

3、 為什么不將key設(shè)置為強(qiáng)引用

3.1 、key 如果是強(qiáng)引用

那么為什么ThreadLocalMap的key要設(shè)計(jì)成弱引用呢?其實(shí)很簡單,如果key設(shè)計(jì)成強(qiáng)引用且沒有手動(dòng)remove(),那么key會(huì)和value一樣伴隨線程的整個(gè)生命周期。

1、假設(shè)在業(yè)務(wù)代碼中使用完ThreadLocal, ThreadLocal ref被回收了,但是因?yàn)閠hreadLocalMap的Entry強(qiáng)引用了threadLocal(key就是threadLocal), 造成ThreadLocal無法被回收。在沒有手動(dòng)刪除Entry以及CurrentThread(當(dāng)前線程)依然運(yùn)行的前提下, 始終有強(qiáng)引用鏈CurrentThread Ref → CurrentThread →Map(ThreadLocalMap)-> entry, Entry就不會(huì)被回收( Entry中包括了ThreadLocal實(shí)例和value), 導(dǎo)致Entry內(nèi)存泄漏也就是說: ThreadLocalMap中的key使用了強(qiáng)引用, 是無法完全避免內(nèi)存泄漏的。請結(jié)合圖1看。

3.2  那么為什么 key 要用弱引用

事實(shí)上,在 ThreadLocalMap 中的set/getEntry 方法中,會(huì)對 key 為 null(也即是 ThreadLocal 為 null )進(jìn)行判斷,如果為 null 的話,那么會(huì)把 value 置為 null 的.這就意味著使用threadLocal , CurrentThread 依然運(yùn)行的前提下.就算忘記調(diào)用 remove 方法,弱引用比強(qiáng)引用可以多一層保障:弱引用的 ThreadLocal 會(huì)被回收.對應(yīng)value在下一次 ThreadLocaI 調(diào)用 get()/set()/remove() 中的任一方法的時(shí)候會(huì)被清除,從而避免內(nèi)存泄漏.

3.3 如何正確的使用ThreadLocal

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

 2、每次使用完ThreadLocal,都調(diào)用它的remove()方法,清除數(shù)據(jù)。

到此這篇關(guān)于淺談Java中ThreadLocal內(nèi)存泄露的原因及處理方式的文章就介紹到這了,更多相關(guān)Java ThreadLocal內(nèi)存泄露內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • SpringBoot使用JDBC獲取相關(guān)的數(shù)據(jù)方法

    SpringBoot使用JDBC獲取相關(guān)的數(shù)據(jù)方法

    這篇文章主要介紹了SpringBoot使用JDBC獲取相關(guān)的數(shù)據(jù)方法,JDBC與數(shù)據(jù)庫建立連接、發(fā)送 操作數(shù)據(jù)庫的語句并處理結(jié)果,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2019-03-03
  • Java實(shí)現(xiàn)圖片驗(yàn)證碼功能

    Java實(shí)現(xiàn)圖片驗(yàn)證碼功能

    這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)圖片驗(yàn)證碼功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-09-09
  • Spring Boot 中實(shí)現(xiàn)跨域的多種方式小結(jié)

    Spring Boot 中實(shí)現(xiàn)跨域的多種方式小結(jié)

    Spring Boot提供了多種方式來實(shí)現(xiàn)跨域請求,開發(fā)者可以根據(jù)具體需求選擇適合的方法,在配置時(shí),要確保不僅考慮安全性,還要兼顧應(yīng)用的靈活性和性能,本文給大家介紹Spring Boot 中實(shí)現(xiàn)跨域的多種方式,感興趣的朋友一起看看吧
    2024-01-01
  • JUC三大輔助類CountDownLatch、CyclicBarrier和Semaphore詳解

    JUC三大輔助類CountDownLatch、CyclicBarrier和Semaphore詳解

    這篇文章主要介紹了JUC三大輔助類CountDownLatch、CyclicBarrier和Semaphore詳解,CountDownLatch 類可以設(shè)置一個(gè)計(jì)數(shù)器,然后通過 countDown 方法來進(jìn)行 減 1 的操作,使用 await 方法等待計(jì)數(shù)器不大于 0,然后繼續(xù)執(zhí)行 await 方法 之后的語句,需要的朋友可以參考下
    2024-01-01
  • Java當(dāng)中讓事務(wù)回滾的幾種方式

    Java當(dāng)中讓事務(wù)回滾的幾種方式

    這篇文章主要給大家介紹了關(guān)于Java當(dāng)中讓事務(wù)回滾的幾種方式, 事務(wù)回滾通常用于在某些操作失敗時(shí)取消之前已執(zhí)行的所有操作,這樣,我們就可以保證數(shù)據(jù)的一致性,需要的朋友可以參考下
    2023-08-08
  • JAVA匿名內(nèi)部類語法分析及實(shí)例詳解

    JAVA匿名內(nèi)部類語法分析及實(shí)例詳解

    這篇文章主要介紹了JAVA匿名內(nèi)部類語法分析及實(shí)例詳解,匿名內(nèi)部類可以使你的代碼更加簡潔,它與局部類很相似,不同的是它沒有類名,如果某個(gè)局部類你只需要用一次,那么你就可以使用匿名內(nèi)部類。對此感興趣的可以了解一下
    2020-07-07
  • 5種必會(huì)的Java異步調(diào)用轉(zhuǎn)同步的方法你會(huì)幾種

    5種必會(huì)的Java異步調(diào)用轉(zhuǎn)同步的方法你會(huì)幾種

    這篇文章主要介紹了5種必會(huì)的Java異步調(diào)用轉(zhuǎn)同步的方法你會(huì)幾種,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-12-12
  • Java歸并排序算法代碼實(shí)現(xiàn)

    Java歸并排序算法代碼實(shí)現(xiàn)

    歸并(Merge)排序法是將兩個(gè)(或兩個(gè)以上)有序表合并成一個(gè)新的有序表,即把待排序序列分為若干個(gè)子序列,每個(gè)子序列是有序的,下面這篇文章主要給大家介紹了關(guān)于Java歸并排序算法的相關(guān)資料,需要的朋友可以參考下
    2024-03-03
  • 最新Java?泛型中的通配符講解

    最新Java?泛型中的通配符講解

    Java的泛型是偽泛型,那是因?yàn)榉盒托畔⒅淮嬖谟诖a編譯階段,在生成的字節(jié)碼中是不包含泛型中的類型信息的,使用泛型的時(shí)候加上類型參數(shù),在編譯器編譯的時(shí)候會(huì)去掉,這個(gè)過程為類型擦除,這篇文章主要介紹了Java?泛型中的通配符,需要的朋友可以參考下
    2022-06-06
  • java基礎(chǔ)檢查和未檢查異常處理詳解

    java基礎(chǔ)檢查和未檢查異常處理詳解

    這篇文章介紹了java基礎(chǔ)中異常的處理,主要講解了java檢查和未檢查異常處理的示例詳解有需要的朋友可以借鑒參考下,希望能夠有所幫助
    2021-10-10

最新評論