淺談Java中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對(duì)應(yīng)于不同的value值,他們之間互不影響。ThreadLocal為每一個(gè)線程都提供了變量的副本,使得每個(gè)線程在某一時(shí)間訪問到的并不是同一個(gè)對(duì)象,這樣就隔離了多個(gè)線程對(duì)數(shù)據(jù)的數(shù)據(jù)共享。
2、ThreadLocal 內(nèi)存泄露的原因
Entry將ThreadLocal作為Key,值作為value保存,它繼承自WeakReference,注意構(gòu)造函數(shù)里的第一行代碼super(k),這意味著ThreadLocal對(duì)象是一個(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 方法刪除對(duì)應(yīng)的 Entry ,就能避免內(nèi)存泄漏。
第二點(diǎn)稍微復(fù)雜一點(diǎn),由于ThreadLocalMap 是 Thread 的一個(gè)屬性,被當(dāng)前線程所引用,所以ThreadLocalMap的生命周期跟 Thread 一樣長(zhǎng)。如果threadlocal變量被回收,那么當(dāng)前線程的threadlocal 變量副本指向的就是key=null, 也即entry(null,value),那這個(gè)entry對(duì)應(yīng)的value永遠(yuǎn)無法訪問到。實(shí)際私用ThreadLocal場(chǎng)景都是采用線程池,而線程池中的線程都是復(fù)用的,這樣就可能導(dǎo)致非常多的entry(null,value)出現(xiàn),從而導(dǎo)致內(nèi)存泄露。
綜上, ThreadLocal 內(nèi)存泄漏的根源是:
由于ThreadLocalMap 的生命周期跟 Thread 一樣長(zhǎng),對(duì)于重復(fù)利用的線程來說,如果沒有手動(dòng)刪除(remove()方法)對(duì)應(yīng) key 就會(huì)導(dǎo)致entry(null,value)的對(duì)象越來越多,從而導(dǎo)致內(nèi)存泄漏.
3、 為什么不將key設(shè)置為強(qiáng)引用
3.1 、key 如果是強(qiáng)引用
那么為什么ThreadLocalMap的key要設(shè)計(jì)成弱引用呢?其實(shí)很簡(jiǎn)單,如果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)存泄漏的。請(qǐng)結(jié)合圖1看。
3.2 那么為什么 key 要用弱引用
事實(shí)上,在 ThreadLocalMap 中的set/getEntry 方法中,會(huì)對(duì) key 為 null(也即是 ThreadLocal 為 null )進(jìn)行判斷,如果為 null 的話,那么會(huì)把 value 置為 null 的.這就意味著使用threadLocal , CurrentThread 依然運(yùn)行的前提下.就算忘記調(diào)用 remove 方法,弱引用比強(qiáng)引用可以多一層保障:弱引用的 ThreadLocal 會(huì)被回收.對(duì)應(yīng)value在下一次 ThreadLocaI 調(diào)用 get()/set()/remove() 中的任一方法的時(shí)候會(huì)被清除,從而避免內(nèi)存泄漏.
3.3 如何正確的使用ThreadLocal
1、將ThreadLocal變量定義成private static的,這樣的話ThreadLocal的生命周期就更長(zhǎng),由于一直存在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)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java8新特性之接口中的默認(rèn)方法和靜態(tài)方法
這篇文章主要介紹了Java8新特性之接口中的默認(rèn)方法和靜態(tài)方法的相關(guān)資料,文中講解非常細(xì)致,代碼幫助大家更好的理解和學(xué)習(xí),感興趣的朋友可以了解下2020-07-07
Spring MVC 簡(jiǎn)單的hello world的實(shí)現(xiàn)
這篇文章主要介紹了Spring MVC 簡(jiǎn)單的hello world的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-01-01
Java中System.currentTimeMillis()計(jì)算方式與時(shí)間單位轉(zhuǎn)換講解
本文詳細(xì)講解了Java中System.currentTimeMillis()計(jì)算方式與時(shí)間單位轉(zhuǎn)換,對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-12-12
springboot的實(shí)體類字段校驗(yàn)的分組校驗(yàn)具體實(shí)現(xiàn)步驟
分組校驗(yàn)允許在不同場(chǎng)景下對(duì)同一實(shí)體類應(yīng)用不同的校驗(yàn)規(guī)則,通過定義分組接口、在實(shí)體類和Controller中指定分組,以及全局異常處理,可以靈活控制校驗(yàn)規(guī)則,本文介紹springboot的實(shí)體類字段校驗(yàn)的分組校驗(yàn),感興趣的朋友一起看看吧2025-03-03
關(guān)于SpringBoot微服務(wù)發(fā)布與部署的三種方式
SpringBoot 框架只提供了一套基于可執(zhí)行 jar 包(executable jar)格式的標(biāo)準(zhǔn)發(fā)布形式,但并沒有對(duì)部署做過多的界定,而且為了簡(jiǎn)化可執(zhí)行 jar 包的生成,SpringBoot 提供了相應(yīng)的 Maven 項(xiàng)目插件,需要的朋友可以參考下2023-05-05

