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

