關(guān)于ThreadLocal的用法和說明及注意事項(xiàng)
ThreadLocal
ThreadLocal是用于解決Java并發(fā)安全性問題的一個(gè)類。
其主要作用是防止不同線程中的數(shù)據(jù)沖突。
原理圖
如下:
原理說明
創(chuàng)建一個(gè)ThreadLocal<V>類的對象,默認(rèn)會在每一個(gè)線程中都開啟一小片區(qū)域,該片區(qū)域可以理解為kay value格式的(實(shí)質(zhì)上是在Thread中有內(nèi)部類ThreadLocalMap,每聲明了一個(gè)ThreadLocal,就相當(dāng)于在這個(gè)ThreadLocalMap中設(shè)置了一個(gè)<key,value>,因?yàn)榫€程是相互獨(dú)立的,所以ThreadLocalMap也是獨(dú)立的),ThreadLocalMap中以ThreadLocal實(shí)例引用的變量名為key,V為value。
每一個(gè)V都是線程獨(dú)有的!
使用
ThreadLocal類接口很簡單,只有4個(gè)方法:
• void set(Object value)
- 設(shè)置當(dāng)前線程的線程局部變量的值。
• public Object get()
- 該方法返回當(dāng)前線程所對應(yīng)的線程局部變量。
• public void remove()
- 將當(dāng)前線程局部變量的值刪除,目的是為了減少內(nèi)存的占用,該方法是JDK 5.0新增的方法。
- 需要指出的是,當(dāng)線程結(jié)束后,對應(yīng)該線程的局部變量將自動被垃圾回收,所以顯式調(diào)用該方法清除線程的局部變量并不是必須的操作,但它可以加快內(nèi)存回收的速度。
• protected Object initialValue()
- 返回該線程局部變量的初始值,該方法是一個(gè)protected的方法,顯然是為了讓子類覆蓋而設(shè)計(jì)的。
- 這個(gè)方法是一個(gè)延遲調(diào)用方法,在線程第1次調(diào)用get()或set(Object)時(shí)才執(zhí)行,并且僅執(zhí)行1次。
- ThreadLocal中的缺省實(shí)現(xiàn)直接返回一個(gè)null。
實(shí)例!
public final static ThreadLocal<String> threadLocal= new ThreadLocal<String>();
threadLocal代表一個(gè)能夠存放String類型的ThreadLocal對象。
此時(shí)不論什么一個(gè)線程能夠并發(fā)訪問這個(gè)變量,對它進(jìn)行寫入、讀取操作,都是線程安全的。
注意?。?!
ThreadLocal如果應(yīng)用不妥當(dāng)會導(dǎo)致內(nèi)存泄漏。
先來說下什么是內(nèi)存泄漏和內(nèi)存溢出,內(nèi)存泄漏是指某個(gè)變量申請了內(nèi)存的資源,但是引用釋放了,這樣就導(dǎo)致占用著內(nèi)存卻不能訪問到(俗話叫占著茅坑不拉屎?。?/p>
內(nèi)存溢出是指某個(gè)變量在申請內(nèi)存空間資源的時(shí)候需要的空間大于實(shí)際的空間,即為內(nèi)存空間不足了(人太多坑不夠了!)
如圖解:
當(dāng)寫下 o=null時(shí),只是表示o不再指向堆中object的對象實(shí)例,不代表這個(gè)對象實(shí)例不存在了。
下面來說明下Java中創(chuàng)建引用的幾種方法
- 強(qiáng)引用就是指在程序代碼之中普遍存在的,類似“Object obj=new Object()”這類的引用,只要強(qiáng)引用還存在,垃圾收集器永遠(yuǎn)不會回收掉被引用的對象實(shí)例。
- 軟引用是用來描述一些還有用但并非必需的對象。對于軟引用關(guān)聯(lián)著的對象,在系統(tǒng)將要發(fā)生內(nèi)存溢出異常之前,將會把這些對象實(shí)例列進(jìn)回收范圍之中進(jìn)行第二次回收。如果這次回收還沒有足夠的內(nèi)存,才會拋出內(nèi)存溢出異常。在JDK 1.2之后,提供了SoftReference類來實(shí)現(xiàn)軟引用。
- 弱引用也是用來描述非必需對象的,但是它的強(qiáng)度比軟引用更弱一些,被弱引用關(guān)聯(lián)的對象實(shí)例只能生存到下一次垃圾收集發(fā)生之前。當(dāng)垃圾收集器工作時(shí),無論當(dāng)前內(nèi)存是否足夠,都會回收掉只被弱引用關(guān)聯(lián)的對象實(shí)例。在JDK 1.2之后,提供了WeakReference類來實(shí)現(xiàn)弱引用。
- 虛引用也稱為幽靈引用或者幻影引用,它是最弱的一種引用關(guān)系。一個(gè)對象實(shí)例是否有虛引用的存在,完全不會對其生存時(shí)間構(gòu)成影響,也無法通過虛引用來取得一個(gè)對象實(shí)例。為一個(gè)對象設(shè)置虛引用關(guān)聯(lián)的唯一目的就是能在這個(gè)對象實(shí)例被收集器回收時(shí)收到一個(gè)系統(tǒng)通知。在JDK 1.2之后,提供了PhantomReference類來實(shí)現(xiàn)虛引用。
這里只舉一個(gè)軟引用的例子:
SoftReference<String> ref = new SoftReference<String>("Hello world");
這樣就設(shè)置了 ref 對內(nèi)存中 "Hello world"的軟引用。
ThreadLocal產(chǎn)生內(nèi)存泄漏的原因
根據(jù)我們前面對ThreadLocal的分析,我們可以知道每個(gè)Thread 擁有一個(gè) ThreadLocalMap,這個(gè)映射表的 key 是 ThreadLocal實(shí)例本身,value 是真正需要存儲的 Object,也就是說 ThreadLocal 本身并不存儲值,它只是作為一個(gè) key 來讓線程從 ThreadLocalMap 獲取 value。
仔細(xì)觀察ThreadLocalMap,這個(gè)map是使用 ThreadLocal 的弱引用作為 Key 的,弱引用的對象在 GC 時(shí)會被回收。
圖中的虛線表示弱引用。
這樣,當(dāng)把threadlocal變量置為null以后,沒有任何強(qiáng)引用指向threadlocal實(shí)例,所以threadlocal將會被gc回收。這樣一來,ThreadLocalMap中就會出現(xiàn)key為null的Entry,就沒有辦法訪問這些key為null的Entry的value,如果當(dāng)前線程再遲遲不結(jié)束的話,這些key為null的Entry的value就會一直存在一條強(qiáng)引用鏈:Thread Ref -> Thread -> ThreaLocalMap -> Entry -> value,而這塊value永遠(yuǎn)不會被訪問到了,所以存在著內(nèi)存泄露。
只有當(dāng)前thread結(jié)束以后,current thread就不會存在棧中,強(qiáng)引用斷開,Current Thread、Map value將全部被GC回收。最好的做法是不在需要使用ThreadLocal變量后,都調(diào)用它的remove()方法,清除數(shù)據(jù)。
所以回到我們前面的實(shí)驗(yàn)場景,場景3中,雖然線程池里面的任務(wù)執(zhí)行完畢了,但是線程池里面的5個(gè)線程會一直存在直到JVM退出,我們set了線程的localVariable變量后沒有調(diào)用localVariable.remove()方法,導(dǎo)致線程池里面的5個(gè)線程的threadLocals變量里面的new LocalVariable()實(shí)例沒有被釋放。
其實(shí)考察ThreadLocal的實(shí)現(xiàn),我們可以看見,無論是get()、set()在某些時(shí)候,調(diào)用了expungeStaleEntry方法用來清除Entry中Key為null的Value,但是這是不及時(shí)的,也不是每次都會執(zhí)行的,所以一些情況下還是會發(fā)生內(nèi)存泄露。只有remove()方法中顯式調(diào)用了expungeStaleEntry方法。
從表面上看內(nèi)存泄漏的根源在于使用了弱引用,但是另一個(gè)問題也同樣值得思考:為什么使用弱引用而不是強(qiáng)引用?
下面我們分兩種情況討論
- key 使用強(qiáng)引用:對ThreadLocal對象實(shí)例的引用被置為null了,但是ThreadLocalMap還持有這個(gè)ThreadLocal對象實(shí)例的強(qiáng)引用,如果沒有手動刪除,ThreadLocal的對象實(shí)例不會被回收,導(dǎo)致Entry內(nèi)存泄漏。
- key 使用弱引用:對ThreadLocal對象實(shí)例的引用被被置為null了,由于ThreadLocalMap持有ThreadLocal的弱引用,即使沒有手動刪除,ThreadLocal的對象實(shí)例也會被回收。value在下一次ThreadLocalMap調(diào)用set,get,remove都有機(jī)會被回收。
比較兩種情況,我們可以發(fā)現(xiàn):
由于ThreadLocalMap的生命周期跟Thread一樣長,如果都沒有手動刪除對應(yīng)key,都會導(dǎo)致內(nèi)存泄漏,但是使用弱引用可以多一層保障。
因此,ThreadLocal內(nèi)存泄漏的根源是:
由于ThreadLocalMap的生命周期跟Thread一樣長,如果沒有手動刪除對應(yīng)key就會導(dǎo)致內(nèi)存泄漏,而不是因?yàn)槿跻谩?/p>
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
ArrayList源碼探秘之Java動態(tài)數(shù)組的實(shí)現(xiàn)
這篇文章將帶大家從ArrayList源碼來探秘一下Java動態(tài)數(shù)組的實(shí)現(xiàn),文中的示例代碼講解詳細(xì),對我們深入了解JavaScript有一定的幫助,需要的可以參考一下2023-08-08Spring Security實(shí)現(xiàn)動態(tài)路由權(quán)限控制方式
這篇文章主要介紹了Spring Security實(shí)現(xiàn)動態(tài)路由權(quán)限控制方式,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-08-08MyBatis-Plus插件機(jī)制及通用Service新功能
這篇文章主要介紹了MyBatis-Plus插件機(jī)制以及通用Service、新功能,本文通過實(shí)例圖文相結(jié)合給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-07-07Spring Boot實(shí)現(xiàn)跨域訪問實(shí)現(xiàn)代碼
本文通過實(shí)例代碼給大家介紹了Spring Boot實(shí)現(xiàn)跨域訪問的知識,然后在文中給大家介紹了spring boot 服務(wù)器端設(shè)置允許跨域訪問 的方法,感興趣的朋友一起看看吧2017-07-07springcloud gateway網(wǎng)關(guān)服務(wù)啟動報(bào)錯(cuò)的解決
這篇文章主要介紹了springcloud gateway網(wǎng)關(guān)服務(wù)啟動報(bào)錯(cuò)的解決,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03java中如何使用BufferedImage判斷圖像通道順序并轉(zhuǎn)RGB/BGR
這篇文章主要介紹了java中如何BufferedImage判斷圖像通道順序并轉(zhuǎn)RGB/BGR的相關(guān)資料,需要的朋友可以參考下2017-03-03