分析ThreadLocal內(nèi)存泄漏問(wèn)題
ThreadLocal的實(shí)現(xiàn)原理
ThreadLocal
的實(shí)現(xiàn):
每一個(gè)Thread
內(nèi)部維護(hù)一個(gè)ThreadLocalMap
映射表,這個(gè)映射表的key
是ThreadLocal
實(shí)例本身,value
是真正需要存儲(chǔ)的Object
。
也就是說(shuō)ThreadLocal
本身不存儲(chǔ)值,它只是作為一個(gè)key
來(lái)讓線(xiàn)程從ThreadLocalMap
獲取value
的。但是ThreadLocalMap
是使用ThreadLocal
的弱引用作為key
的,弱引用的對(duì)象在GC
時(shí)會(huì)被回收。
ThreadLocal為什么會(huì)內(nèi)存泄漏
ThreadLocalMap
使用ThreadLocal
的弱引用作為key
,如果一個(gè)ThreadLocal
沒(méi)有外部強(qiáng)引用來(lái)引用它,那么系統(tǒng)GC
的時(shí)候,這個(gè)ThreadLocal
會(huì)被回收,這樣一來(lái),ThreadLocalMap
中會(huì)出現(xiàn)key
為null
的Entry
,這樣就沒(méi)有辦法訪問(wèn)key
為null
的Entry
的value
,如果當(dāng)前線(xiàn)程遲遲不結(jié)束,這些key
為null
的Entry
的value
就會(huì)存在一條強(qiáng)引用鏈,永遠(yuǎn)無(wú)法回收,造成內(nèi)存泄漏。
其實(shí)ThreadLocal
的設(shè)計(jì)中已經(jīng)考慮到了這種情況,也加上了一些預(yù)防措施,在調(diào)用get
、set
、remove
方法的時(shí)候,會(huì)清楚線(xiàn)程ThreadLocalMap
里所有key
為null
的value
。
但是這些被動(dòng)的預(yù)防措施并不能保證不會(huì)內(nèi)存泄漏:
- 使用static的ThreadLocal,延長(zhǎng)了ThreadLocal的生命周期,可能導(dǎo)致的內(nèi)存泄漏。
- 分配使用了ThreadLocal又不再調(diào)用get() ,set() ,remove() 方法,那么就會(huì)導(dǎo)致內(nèi)存泄漏。
為什么使用弱引用
從表面上看內(nèi)存泄漏的根本原因是使用了弱引用,那么為什么使用弱引用而不使用強(qiáng)引用呢?下面看看官方文檔的說(shuō)法:
To help deal with very large and long-lived usages, the hash table entries use WeakReferences for keys.
翻譯過(guò)來(lái)就是:為了應(yīng)對(duì)非常大和長(zhǎng)時(shí)間的用途,哈希表使用弱引用。
下面我們分兩種情況討論:
- key 使用強(qiáng)引用:引用的ThreadLocal的對(duì)象被回收了,但是ThreadLocalMap還持有ThreadLocal的強(qiáng)引用,如果沒(méi)有手動(dòng)刪除,ThreadLocal不會(huì)被回收,導(dǎo)致Entry內(nèi)存泄漏。
- key 使用弱引用:引用的ThreadLocal的對(duì)象被回收了,由于ThreadLocalMap持有ThreadLocal的弱引用,即使沒(méi)有手動(dòng)刪除,ThreadLocal也會(huì)被回收。value在下一次ThreadLocalMap調(diào)用set,get,remove的時(shí)候會(huì)被清除。
比較兩種情況,我們可以發(fā)現(xiàn):由于ThreadLocalMap的生命周期跟Thread一樣長(zhǎng),如果都沒(méi)有手動(dòng)刪除對(duì)應(yīng)key,都會(huì)導(dǎo)致內(nèi)存泄漏,但是使用弱引用可以多一層保障:弱引用 ThreadLocal 不會(huì)內(nèi)存泄漏,對(duì)應(yīng)的 value 在下一次 ThreadLocalMap 調(diào)用 set , get , remove 的時(shí)候會(huì)被清除。
因此,ThreadLocal內(nèi)存泄漏的根源是:由于ThreadLocalMap的生命周期跟Thread一樣長(zhǎng),如果沒(méi)有手動(dòng)刪除對(duì)應(yīng)key就會(huì)導(dǎo)致內(nèi)存泄漏,而不是因?yàn)槿跻谩?/p>
ThreadLocal最佳實(shí)踐
綜合上面的分析,我們可以理解ThreadLocal內(nèi)存泄漏的前因后果,那么怎么避免內(nèi)存泄漏呢?
- 每次使用完ThreadLocal,都調(diào)用它的remove() 方法,清除數(shù)據(jù)。
在使用線(xiàn)程池的情況下,沒(méi)有及時(shí)清理ThreadLocal,不僅是內(nèi)存泄漏的問(wèn)題,更嚴(yán)重的是可能導(dǎo)致業(yè)務(wù)邏輯出現(xiàn)問(wèn)題。所以,使用ThreadLocal就跟加鎖完要解鎖一樣,用完就清理。
到此這篇關(guān)于分析ThreadLocal內(nèi)存泄漏問(wèn)題的文章就介紹到這了,更多相關(guān)ThreadLocal內(nèi)存泄漏內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
IntelliJ IDEA Project窗口的一些設(shè)置詳解
這篇文章主要介紹了IntelliJ IDEA Project窗口的一些設(shè)置詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08使用spring boot開(kāi)發(fā)時(shí)java對(duì)象和Json對(duì)象轉(zhuǎn)換的問(wèn)題
這篇文章主要介紹了使用spring boot開(kāi)發(fā)時(shí)java對(duì)象和Json對(duì)象轉(zhuǎn)換的問(wèn)題,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-03-03手把手教你SpringBoot快速集成Swagger的配置過(guò)程
這篇文章主要介紹了手把手教你SpringBoot快速集成Swagger的配置過(guò)程,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-02-02java 打印一字符串,并在main()方法內(nèi)調(diào)用它
編寫(xiě)一個(gè)方法(名字自定,但要符合Java編碼規(guī)范),方法內(nèi)打印一字符串,并在main()方法內(nèi)調(diào)用它。2017-02-02解決rror updating database.Cause:java.sql.SQLSyntaxE
這篇文章主要介紹了解決rror updating database.Cause:java.sql.SQLSyntaxErrorException問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-05-05SpringSceurity實(shí)現(xiàn)短信驗(yàn)證碼功能的示例代碼
這篇文章主要介紹了SpringSceurity實(shí)現(xiàn)短信驗(yàn)證碼功能的示例代碼,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-06-06JAVA中使用JSON進(jìn)行數(shù)據(jù)傳遞示例
本篇文章主要介紹了JAVA中使用JSON進(jìn)行數(shù)據(jù)傳遞示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-01-01Android內(nèi)存泄漏實(shí)戰(zhàn)解析
Java是垃圾回收語(yǔ)言的一種。這篇文章主要介紹了Android內(nèi)存泄漏 的相關(guān)資料,需要的朋友可以參考下2016-10-10