本篇文章給大家分享的是有關(guān)如何探究ThreadLocal內(nèi)存泄漏的原因,小編覺得挺實(shí)用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。
創(chuàng)新互聯(lián)建站-成都網(wǎng)站建設(shè)公司,專注做網(wǎng)站、成都網(wǎng)站設(shè)計(jì)、網(wǎng)站營銷推廣,域名與空間,網(wǎng)站空間,成都網(wǎng)站托管有關(guān)企業(yè)網(wǎng)站制作方案、改版、費(fèi)用等問題,請(qǐng)聯(lián)系創(chuàng)新互聯(lián)建站。
1、首先看下ThreadLocal的原理圖:
在ThreadLocal的生命周期中,都存在這些引用。

其中,實(shí)線代表強(qiáng)引用,虛線代表弱引用;
2、ThreadLocal的實(shí)現(xiàn):每個(gè)Thread維護(hù)一個(gè)ThreadLocalMap映射表,這個(gè)映射表的key是ThreadLocal實(shí)例本身,value是真正需要存儲(chǔ)的Object;
3、也就是說ThreadLocal本身不存儲(chǔ)值,它只是作為一個(gè)key來讓線程從ThreadLocalMap獲取value,值得注意的是圖中的虛線,表示ThreadLocalMap是使用ThreadLocal的弱引用作為key,其在GC時(shí)會(huì)被回收;
4、ThreadLocalMap使用ThreadLocal的弱引用作為key,如果一個(gè)ThreadLocal沒有外部強(qiáng)引用來引用它,那么系統(tǒng) GC 的時(shí)候,這個(gè)ThreadLocal勢(shì)必會(huì)被回收,這樣一來,ThreadLocalMap中就會(huì)出現(xiàn)key為null的Entry,就沒有辦法訪問這些key為null的Entry的value,如果當(dāng)前線程再遲遲不結(jié)束的話,這些key為null的Entry的value就會(huì)一直存在一條強(qiáng)引用鏈:Thread Ref -> Thread -> ThreaLocalMap -> Entry -> value永遠(yuǎn)無法回收,造成內(nèi)存泄漏。
5、總的來說就是,ThreadLocal里面使用了一個(gè)存在弱引用的map, map的類型是ThreadLocal.ThreadLocalMap. Map中的key為一個(gè)threadlocal實(shí)例。這個(gè)Map的確使用了弱引用,不過弱引用只是針對(duì)key。每個(gè)key都弱引用指向threadlocal。當(dāng)把threadlocal實(shí)例置為null以后,沒有任何強(qiáng)引用指向threadlocal實(shí)例,所以threadlocal將會(huì)被gc回收。
但是,我們的value卻不能回收,而這塊value永遠(yuǎn)不會(huì)被訪問到了,所以存在著內(nèi)存泄露。因?yàn)榇嬖谝粭l從current thread連接過來的強(qiáng)引用。只有當(dāng)前thread結(jié)束以后,current thread就不會(huì)存在棧中,強(qiáng)引用斷開,Current Thread、Map value將全部被GC回收。最好的做法是將調(diào)用threadlocal的remove方法,這也是等會(huì)后邊要說的。
6、其實(shí),ThreadLocalMap的設(shè)計(jì)中已經(jīng)考慮到這種情況,也加上了一些防護(hù)措施:在ThreadLocal的get(),set(),remove()的時(shí)候都會(huì)清除線程ThreadLocalMap里所有key為null的value。這一點(diǎn)在上一節(jié)中也講到過!
7、但是這些被動(dòng)的預(yù)防措施并不能保證不會(huì)內(nèi)存泄漏:
a、使用static的ThreadLocal,延長ThreadLocal的生命周期,可能導(dǎo)致內(nèi)存泄漏;
b、分配只用了ThreadLocal又不再調(diào)用get()、set()、remove()方法,那么可能導(dǎo)致內(nèi)存泄漏,因?yàn)檫@塊內(nèi)存會(huì)一直存在;
以下是源碼:
/**
* The entries in this hash map extend WeakReference, using
* its main ref field as the key (which is always a
* ThreadLocal object). Note that null keys (i.e. entry.get()
* == null) mean that the key is no longer referenced, so the
* entry can be expunged from table. Such entries are referred to
* as "stale entries" in the code that follows.
*/
static class Entry extends WeakReference<ThreadLocal<?>> {
/** The value associated with this ThreadLocal. */
Object value;
Entry(ThreadLocal<?> k, Object v) {
super(k);
value = v;
}
}以上就是如何探究ThreadLocal內(nèi)存泄漏的原因,小編相信有部分知識(shí)點(diǎn)可能是我們?nèi)粘9ぷ鲿?huì)見到或用到的。希望你能通過這篇文章學(xué)到更多知識(shí)。更多詳情敬請(qǐng)關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。
名稱欄目:如何探究ThreadLocal內(nèi)存泄漏的原因
分享路徑:http://www.chinadenli.net/article24/gjogje.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供微信公眾號(hào)、營銷型網(wǎng)站建設(shè)、用戶體驗(yàn)、關(guān)鍵詞優(yōu)化、靜態(tài)網(wǎng)站、網(wǎng)站設(shè)計(jì)公司
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)