ThreadLocal 內(nèi)存泄露的原因及處理方式
目錄
1、ThreadLocal 使用原理
2、ThreadLocal 內(nèi)存泄露的原因
3、 為什么不將key設置為強引用
3.1 、key 如果是強引用
3.2、key 如果是強引用
3.3??那么為什么 key 要用弱引用
3.4 如何正確的使用ThreadLocal
1、ThreadLocal 使用原理
? ? ? ?前文我們講過ThreadLocal的主要用途是實現(xiàn)線程間變量的隔離,表面上他們使用的是同一個ThreadLocal, 但是實際上使用的值value卻是自己獨有的一份。用一圖直接表示threadlocal 的使用方式。
圖1
從圖中我們可以當線程使用threadlocal 時,是將threadlocal當做當前線程thread的屬性ThreadLocalMap 中的一個Entry的key值,實際上存放的變量是Entry的value值,我們實際要使用的值是value值。value值為什么不存在并發(fā)問題呢,因為它只有一個線程能訪問。threadlocal我們可以當做一個索引看待,可以有多個threadlocal 變量,不同的threadlocal對應于不同的value值,他們之間互不影響。ThreadLocal為每一個線程都提供了變量的副本,使得每個線程在某一時間訪問到的并不是同一個對象,這樣就隔離了多個線程對數(shù)據(jù)的數(shù)據(jù)共享。
2、ThreadLocal 內(nèi)存泄露的原因 ?
?Entry將ThreadLocal作為Key,值作為value保存,它繼承自WeakReference,注意構(gòu)造函數(shù)里的第一行代碼super(k),這意味著ThreadLocal對象是一個「弱引用」??梢钥磮D1.
static class Entry extends WeakReference<ThreadLocal<?>> {
/** The value associated with this ThreadLocal. */
Object value;
Entry(ThreadLocal<?> k, Object v) {
super(k);
value = v;
}
}
主要兩個原因
1 . 沒有手動刪除這個 Entry
2 . CurrentThread 當前線程依然運行
????????第一點很好理解,只要在使用完下 ThreadLocal ,調(diào)用其 remove 方法刪除對應的 Entry ,就能避免內(nèi)存泄漏。
????????第二點稍微復雜一點,由于ThreadLocalMap 是 Thread 的一個屬性,被當前線程所引用,所以ThreadLocalMap的生命周期跟 Thread 一樣長。如果threadlocal變量被回收,那么當前線程的threadlocal 變量副本指向的就是key=null, 也即entry(null,value),那這個entry對應的value永遠無法訪問到。實際私用ThreadLocal場景都是采用線程池,而線程池中的線程都是復用的,這樣就可能導致非常多的entry(null,value)出現(xiàn),從而導致內(nèi)存泄露。
綜上, ThreadLocal 內(nèi)存泄漏的根源是:
? ? 由于ThreadLocalMap 的生命周期跟 Thread 一樣長,對于重復利用的線程來說,如果沒有手動刪除(remove()方法)對應 key 就會導致entry(null,value)的對象越來越多,從而導致內(nèi)存泄漏.
3、 為什么不將key設置為強引用
3.1 、key 如果是強引用
? ? ?那么為什么ThreadLocalMap的key要設計成弱引用呢?其實很簡單,如果key設計成強引用且沒有手動remove(),那么key會和value一樣伴隨線程的整個生命周期。
? ?1、假設在業(yè)務代碼中使用完ThreadLocal, ThreadLocal ref被回收了,但是因為threadLocalMap的Entry強引用了threadLocal(key就是threadLocal), 造成ThreadLocal無法被回收。在沒有手動刪除Entry以及CurrentThread(當前線程)依然運行的前提下, 始終有強引用鏈CurrentThread Ref → CurrentThread →Map(ThreadLocalMap)-> entry, Entry就不會被回收( Entry中包括了ThreadLocal實例和value), 導致Entry內(nèi)存泄漏也就是說: ThreadLocalMap中的key使用了強引用, 是無法完全避免內(nèi)存泄漏的。請結(jié)合圖1看。
3.3??那么為什么 key 要用弱引用
? ? ?事實上,在 ThreadLocalMap 中的set/getEntry 方法中,會對 key 為 null(也即是 ThreadLocal 為 null )進行判斷,如果為 null 的話,那么會把 value 置為 null 的.這就意味著使用threadLocal , CurrentThread 依然運行的前提下.就算忘記調(diào)用 remove 方法,弱引用比強引用可以多一層保障:弱引用的 ThreadLocal 會被回收.對應value在下一次 ThreadLocaI 調(diào)用 get()/set()/remove() 中的任一方法的時候會被清除,從而避免內(nèi)存泄漏.
?
3.4 如何正確的使用ThreadLocal
?1、將ThreadLocal變量定義成private static的,這樣的話ThreadLocal的生命周期就更長,由于一直存在ThreadLocal的強引用,所以ThreadLocal也就不會被回收,也就能保證任何時候都能根據(jù)ThreadLocal的弱引用訪問到Entry的value值,然后remove它,防止內(nèi)存泄露文章來源:http://www.zghlxwxcb.cn/news/detail-786745.html
?2、每次使用完ThreadLocal,都調(diào)用它的remove()方法,清除數(shù)據(jù)。
?文章來源地址http://www.zghlxwxcb.cn/news/detail-786745.html
到了這里,關于史上最全ThreadLocal 詳解(二)的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關文章,希望大家以后多多支持TOY模板網(wǎng)!