ThreadLocal(中)實(shí)現(xiàn)原理

ThreadLocal實(shí)現(xiàn)原理

下面是ThreadLocal相關(guān)類的類結(jié)構(gòu)圖逞壁,如圖:

image-20200407164556036.png

由該圖可知流济,Thread類中有一個(gè)threadLocals和一個(gè)inheritableThreadLocals,它們都是ThreadLocalMap類型的變量猾担,而ThreadLocalMap是一個(gè)定制化的HashMap袭灯。在默認(rèn)情況下,每個(gè)線程中的這兩個(gè)變量都為null绑嘹,只有當(dāng)線程第一次調(diào)用ThreadLocal的set()或get()方法時(shí)才華創(chuàng)建它們稽荧。其實(shí)每個(gè)線程的本地變量不是存放在ThreadLocal實(shí)例里面,而是存放在具體線程內(nèi)存空間中。ThreadLocal就是一個(gè)工具殼姨丈,它通過set方法把value值放入調(diào)用線程的threadLocals里面并存放起來畅卓,當(dāng)調(diào)用線程調(diào)用它的get方法時(shí),再從當(dāng)前線程的threadLocals變量里面將其拿出來使用蟋恬。如果調(diào)用線程一直不重質(zhì)翁潘,那么這個(gè)本地變量會一直存放在調(diào)用線程的threadLocals變量里面,所以當(dāng)不需要使用本地變量的時(shí)候可以通過調(diào)用ThreadLocal變量的remove()方法歼争,從當(dāng)前線程的threadLocals里面刪除該本地變量拜马。另外,Thread里面的threadLocals為何被設(shè)計(jì)為map結(jié)構(gòu)沐绒?很明顯是因?yàn)槊總€(gè)線程可以慣量多個(gè)ThreadLocal變量俩莽。

下面簡單分析ThreadLocal的set,get以及remove方法的實(shí)現(xiàn)邏輯乔遮。

void set(T value)
public void set(T value) {
    // (1) 獲取當(dāng)前線程
    Thread t = Thread.currentThread();
    // (2) 將當(dāng)前線程作為key扮超,去查找對應(yīng)的線程變量,找到則設(shè)置
    ThreadLocalMap map = getMap(t);
    if (map != null) 
        map.set(this, value);
    else
        // (3) 第一次調(diào)用就創(chuàng)建當(dāng)前線程對應(yīng)的HashMap
        createMap(t, value);
}

代碼 (1) 首先獲取調(diào)用線程蹋肮,然后使用當(dāng)前線程作為參數(shù)調(diào)用getMap(t)方法出刷,getMap(Thread t)的代碼如下。

ThreadLocalMap getMap(Thread t) {
    return t.threadLocals;
}

可以看到坯辩,getMap(t)的作用時(shí)獲取線程自己的變量threadLocals馁龟,threadLocal變量被綁定到了線程的成員變量上。

如果getMap(t)的返回值不為空漆魔,則把value值設(shè)置到threadLocals中屁柏,也就是把當(dāng)前變量放入當(dāng)前線程的內(nèi)存變量threadLocals中。threadLocals是一個(gè)HashMap結(jié)構(gòu)有送,其中key就是當(dāng)前ThreadLocal的實(shí)例對象引用,value是通過set方法傳遞值僧家。

如果getMap(t)返回值為空則說明是第一次調(diào)用set方法雀摘,這時(shí)創(chuàng)建當(dāng)前線程的threadLocals變量。下面來看createMap(t, value)做什么八拱。

void createMap(Thread t, T firstValue) {
    t.threadLocals = new ThreadLocalMap(this, firstValue);
}

它創(chuàng)建當(dāng)前線程的threadLocals變量阵赠。

T get()
public T get() {
    // (1) 獲取當(dāng)前線程
    Thread t = Thread.currentThread();
    // (2) 獲取當(dāng)前線程的threadLocals變量
    ThreadLocalMap map = getMap(t);
    // (3) 如果threadLocals不為null,則返回對應(yīng)本地變量的值
    if (map != null) {
        ThreadLocalMap.Entity e = map.getEntity(this);
        if (e != null) {
            @SuppressWrarnings("unchecked")
            T result = (T)e.value;
            return result;
        }
    }
    
    // (4) threadLocals為空則初始化當(dāng)前線程的threadLocals成員變量
    return setInitialValue();
    
}

代碼 (1) 首先獲取當(dāng)前線程實(shí)例肌稻,如果當(dāng)前線程的threadLocals變量不為null清蚀,則直接返回當(dāng)前線程綁定的本地線程變量,負(fù)責(zé)執(zhí)行代碼 (4) 進(jìn)行初始化爹谭。setInitialValue() 的代碼如下枷邪。

private T setInitialValue() {
    // (5) 初始化為null
    T value = initialValue();
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    // (6) 如果當(dāng)前線程的threadLocals不為空
    if (map != null)
        map.set(this, value);
    else
    // (7) 如果當(dāng)前線程的threadLocals變量為空
        createMap(t, value);
    return value;
}


protected T initialValue() {
    return null;
}

如果當(dāng)前線程的threadLocals變量不為空,則設(shè)置當(dāng)前線程的本地線程變量值為null诺凡,否則調(diào)用createMap方法創(chuàng)建當(dāng)前線程的createMap變量东揣。

void remove()
public void remove() {
    ThreadLocalMap m = getMap(Thread.currentThread());
    if (m != null)
        m.remove(this);
}

以上代碼所示践惑,如果當(dāng)前線程的threadLocals變量不為空,則刪除當(dāng)前線程中指定ThreadLocal實(shí)例的本地變量嘶卧《酰‘

總結(jié):

在每個(gè)線程內(nèi)部都有一個(gè)名為threadLocals的成員變量,該變量的類型為HashMap芥吟,其中key為我們定義的ThreadLocal變量的this引用侦铜,value則為我們使用set方法設(shè)置的值。 每個(gè)線程本地變量存放在線程自己的內(nèi)存變量threadLocals中钟鸵,如果當(dāng)前線程一直不消亡钉稍,那么這些本地變量會一直存在,所以可能會造成內(nèi)存溢出携添,因此使用完畢后要記得調(diào)用ThreadLocal的remove方法刪除對應(yīng)線程的threadLocals中的本地變量嫁盲。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市烈掠,隨后出現(xiàn)的幾起案子羞秤,更是在濱河造成了極大的恐慌,老刑警劉巖左敌,帶你破解...
    沈念sama閱讀 212,383評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件瘾蛋,死亡現(xiàn)場離奇詭異,居然都是意外死亡矫限,警方通過查閱死者的電腦和手機(jī)哺哼,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,522評論 3 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來叼风,“玉大人取董,你說我怎么就攤上這事∥匏蓿” “怎么了茵汰?”我有些...
    開封第一講書人閱讀 157,852評論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長孽鸡。 經(jīng)常有香客問我蹂午,道長,這世上最難降的妖魔是什么彬碱? 我笑而不...
    開封第一講書人閱讀 56,621評論 1 284
  • 正文 為了忘掉前任豆胸,我火速辦了婚禮,結(jié)果婚禮上巷疼,老公的妹妹穿的比我還像新娘晚胡。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,741評論 6 386
  • 文/花漫 我一把揭開白布搬泥。 她就那樣靜靜地躺著桑寨,像睡著了一般。 火紅的嫁衣襯著肌膚如雪忿檩。 梳的紋絲不亂的頭發(fā)上尉尾,一...
    開封第一講書人閱讀 49,929評論 1 290
  • 那天,我揣著相機(jī)與錄音燥透,去河邊找鬼沙咏。 笑死,一個(gè)胖子當(dāng)著我的面吹牛班套,可吹牛的內(nèi)容都是我干的肢藐。 我是一名探鬼主播,決...
    沈念sama閱讀 39,076評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼吱韭,長吁一口氣:“原來是場噩夢啊……” “哼吆豹!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起理盆,我...
    開封第一講書人閱讀 37,803評論 0 268
  • 序言:老撾萬榮一對情侶失蹤痘煤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后猿规,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體衷快,經(jīng)...
    沈念sama閱讀 44,265評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,582評論 2 327
  • 正文 我和宋清朗相戀三年姨俩,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了蘸拔。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,716評論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡环葵,死狀恐怖调窍,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情张遭,我是刑警寧澤陨晶,帶...
    沈念sama閱讀 34,395評論 4 333
  • 正文 年R本政府宣布,位于F島的核電站帝璧,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏湿刽。R本人自食惡果不足惜的烁,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,039評論 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望诈闺。 院中可真熱鬧渴庆,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,798評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至耸弄,卻和暖如春咧虎,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背计呈。 一陣腳步聲響...
    開封第一講書人閱讀 32,027評論 1 266
  • 我被黑心中介騙來泰國打工砰诵, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人捌显。 一個(gè)月前我還...
    沈念sama閱讀 46,488評論 2 361
  • 正文 我出身青樓茁彭,卻偏偏與公主長得像,于是被迫代替她去往敵國和親扶歪。 傳聞我的和親對象是個(gè)殘疾皇子理肺,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,612評論 2 350

推薦閱讀更多精彩內(nèi)容