并發(fā)編程模式之ThreadLocal源碼和圖文解讀
從一道面試題開始吧,ThreadLocal使用需要注意什么,或者有什么問題?
- 答:如果是在線程池中使用,會存在 1、內(nèi)存泄漏 2、臟數(shù)據(jù) 的問題
- 解決:try finally中調(diào)用 remove方法
慢慢從ThreadLocal的設(shè)計和源碼開始分析,
一、ThreadLocal結(jié)構(gòu)和存取數(shù)據(jù)
當(dāng)創(chuàng)建了一個ThreadLocal對象時,可以將存放的Object對象、或者回調(diào)函數(shù)(延遲加載)放入setInitialValue中;當(dāng)當(dāng)前線程去獲取時,則會在當(dāng)前線程Thread的屬性threadLocals中獲取,而該屬性的類型則為TheadLocal的靜態(tài)內(nèi)部類ThreadLocalMap,但是引用還是指向了Thread;而該threadLocals的ThreadLocalMap內(nèi)部維護了一個其內(nèi)部類ThreadLocal.ThreadLocalMap.Entry數(shù)組,而Entry由key和value組成,key即為當(dāng)前的 new 的ThreadLocal對象本身,value為我們當(dāng)前存儲的Object對象,并且key為WeakReference(弱引用)類型。
所以,ThreadLocal本身只定義了一些內(nèi)部類,而并不真實擁有任何數(shù)據(jù),可以理解為全是空殼子;當(dāng)獲取數(shù)據(jù)時若當(dāng)前線程的ThreadLocal對象(內(nèi)存地址)不存在,則才會在該對象的setInitialValue中copy一份值到Thread的屬性threadLocals中,key為當(dāng)前對象引用,value為存儲的Object值;當(dāng)創(chuàng)建了多個ThreadLocal對象時,當(dāng)每當(dāng)前線程都調(diào)用過ThreadLocal進行get或set時,則會在當(dāng)前線程的threadLocals中存儲多個值,
如下圖:
1、ThreadLocal#get
public T get() { Thread var1 = Thread.currentThread(); ThreadLocal.ThreadLocalMap var2 = this.getMap(var1); if (var2 != null) { ThreadLocal.ThreadLocalMap.Entry var3 = var2.getEntry(this); if (var3 != null) { Object var4 = var3.value; return var4; } } return this.setInitialValue(); }
獲取當(dāng)前的線程,并且使用當(dāng)前線程去獲取ThreadLocal.ThreadLocalMap類型的對象,先看看getMap方法,
ThreadLocal.ThreadLocalMap getMap(Thread var1) { return var1.threadLocals; }
什么都沒有做,只是將傳入的當(dāng)前對象的threadLocals屬性返回,只是該引用還是指向了Thread本身。
繼續(xù),
1、如果threadLocals對象本身不為null,則通過this,即當(dāng)前ThreadLocal對象的引用為key,獲取對應(yīng)Entry的值返回。
2、如果threadLocals對象本身為null,說明當(dāng)前線程中沒有存放過任何ThreadLocal對象的引入的值。
則需要調(diào)setInitialValue方法,如下:
private T setInitialValue() { Object var1 = this.initialValue(); Thread var2 = Thread.currentThread(); ThreadLocal.ThreadLocalMap var3 = this.getMap(var2); if (var3 != null) { var3.set(this, var1); } else { this.createMap(var2, var1); } return var1; }
調(diào)用當(dāng)前ThreadLocal對象的initialValue方法,如果我們沒有重寫,則當(dāng)前默認(rèn)返回null,否則調(diào)用我們重寫的方法,可以理解為懶加載。
每個線程第一次都會調(diào)用該方法獲取的返回值,那么如果沒次調(diào)用都是返回同一個對象則Thread中存儲的就是同一個對象,需要我們自己注意(如果同一對象線程A改變之后線程B也改變了),并且也存在線程安全的問題。
還是獲取當(dāng)前Map是否為null,則調(diào)用createMap方法。
如下:
void createMap(Thread var1, T var2) { var1.threadLocals = new ThreadLocal.ThreadLocalMap(this, var2); }
2、ThreadLocal#set
理解完get的過程,set的就比較容易了,主要是引用本身比較繞。
public void set(T var1) { Thread var2 = Thread.currentThread(); ThreadLocal.ThreadLocalMap var3 = this.getMap(var2); if (var3 != null) { var3.set(this, var1); } else { this.createMap(var2, var1); } }
獲取到Thread的屬性threadLocals,如果是null則new一個,否則就往Map中放一個Entry對象。
二、梳理存在的問題和解決
當(dāng)在線程池中使用的時候,其實大部分情況下我們都會在線程池中使用,比如Tomcat線程池。則key為弱引用,在Root gc不可達的情況下,則會被JVM進行回收。但是正是因為如此value值將永遠的游離,Root gc永遠指向不到該value值,則不能進行g(shù)c。當(dāng)頻繁的調(diào)用,則這樣的對象越來越多,發(fā)生內(nèi)存泄漏。如果該對象的內(nèi)存還比較大,則會照成后續(xù)分配內(nèi)存時新生代不足,則可能照成溢出的情況(溢出的個人理解)。
并且,當(dāng)循環(huán)使用線程的話,比如存放的是用戶信息,如果上一個用戶請求離開時未清除數(shù)據(jù),下一個用戶進來直接獲取的話會拿到上一個用戶的數(shù)據(jù),如果是存儲的積分等,則完全就是臟數(shù)據(jù)。
一并解決上面的兩個問題方法比較簡單,每次在調(diào)用代碼時增加 try finally中調(diào)用 ThreadLocal對象的remove方法,如下:
ThreadLocal#remove
public void remove() { ThreadLocal.ThreadLocalMap var1 = this.getMap(Thread.currentThread()); if (var1 != null) { var1.remove(this); } }
首先還是獲取當(dāng)前的線程去獲取 Thread的屬性threadLocals調(diào)用其(ThreadLocalMap的)remove方法,如下:
private void remove(ThreadLocal<?> var1) { ThreadLocal.ThreadLocalMap.Entry[] var2 = this.table; int var3 = var2.length; int var4 = var1.threadLocalHashCode & var3 - 1; for(ThreadLocal.ThreadLocalMap.Entry var5 = var2[var4]; var5 != null; var5 = var2[var4 = nextIndex(var4, var3)]) { if (var5.get() == var1) { var5.clear(); this.expungeStaleEntry(var4); return; } } }
根據(jù)當(dāng)前的ThreadLocal引用,去Map中循環(huán)匹配,當(dāng)匹配到之后,調(diào)用Entry的remove方法,其實是調(diào)用Refrence的clear方法。最后調(diào)用expungeStaleEntry方法將其對應(yīng)的Entry從數(shù)組中消除。
再看看Reference的clear方法:
public void clear() { this.referent = null; }
將該值直接置位null,則后續(xù)JVM會對該對象進行g(shù)c。
總結(jié)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java Spring Boot實戰(zhàn)練習(xí)之單元測試篇
單元測試(unit testing),是指對軟件中的最小可測試單元進行檢查和驗證。對于單元測試中單元的含義,一般來說,要根據(jù)實際情況去判定其具體含義,如C語言中單元指一個函數(shù),Java里單元指一個類,圖形化的軟件中可以指一個窗口或一個菜單等2021-10-10springBoot下實現(xiàn)java自動創(chuàng)建數(shù)據(jù)庫表
這篇文章主要介紹了springBoot下實現(xiàn)java自動創(chuàng)建數(shù)據(jù)庫表的操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07SpringCloud?Gateway之請求應(yīng)答日志打印方式
這篇文章主要介紹了SpringCloud?Gateway之請求應(yīng)答日志打印方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-03-03win10系統(tǒng)64位jdk1.8的下載與安裝教程圖解
這篇文章主要介紹了win10系統(tǒng)64位jdk1.8的下載與安裝教程圖解,本文給大家介紹的非常詳細,對大家的工作或?qū)W習(xí)具有一定的參考借鑒價值,需要的朋友可以參考下2020-03-03Springboot使用redis實現(xiàn)接口Api限流的實例
本文介紹的內(nèi)容如題,就是利用redis實現(xiàn)接口的限流(某時間范圍內(nèi),最大的訪問次數(shù)),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-07-07詳解Spring Cloud Zuul 服務(wù)網(wǎng)關(guān)
本篇文章主要介紹了詳解Spring Cloud Zuul 服務(wù)網(wǎng)關(guān),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-12-12Java 反轉(zhuǎn)帶頭結(jié)點的單鏈表并顯示輸出的實現(xiàn)過程
這篇文章主要介紹了Java 反轉(zhuǎn)帶頭結(jié)點的單鏈表并顯示輸出,本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-11-11Java直接內(nèi)存和堆內(nèi)存的關(guān)系
在Java編程中,內(nèi)存管理是一個重要的話題,本文介紹了Java中兩種主要內(nèi)存類型:堆內(nèi)存和直接內(nèi)存,堆內(nèi)存是JVM管理的主要內(nèi)存區(qū)域,感興趣的朋友跟隨小編一起看看吧2024-09-09