Java ThreadLocal 實現(xiàn)原理

ThreadLocal 線程本地變量技健,算是Java開發(fā)中比較常用的API了惰拱,今天我們來一探究竟

使用場景

ThreadLocal 適用于每個線程需要自己獨立的實例且該實例需要在多個方法中被使用啊送,也就是變量在線程間隔離馋没,而在同一線程共享的場景降传。例如管理Connection婆排,我們希望每個線程只使用一個Connection實例,這個時候用ThreadLocal就很合適腮猖。

public class ThreadLocalDemo {
    private static final ThreadLocal<Object> threadLocal = new ThreadLocal<>();

    public static void main(String[] args) {
        threadLocal.set(new Object());
        someMethod();
    }

    static void someMethod() {
        // 獲取在threadLocal中存儲的對象
        threadLocal.get();
        
        // 清除ThreadLocal中數(shù)據(jù)
        threadLocal.remove();
    }
}

還有之前寫過的一篇動態(tài)切換數(shù)據(jù)源 http://www.reibang.com/p/0a485c965b8b澈缺,AOP 通過 ThreadLocal 保存當(dāng)前線程需要訪問的數(shù)據(jù)源的key姐赡,AbstractRoutingDataSource 再通過 ThreadLocal 中的數(shù)據(jù)切換到指定的數(shù)據(jù)源柠掂,對業(yè)務(wù)代碼毫無入侵

原理

在我們了解了如何使用之后涯贞,來看下 ThreadLocal 是如何實現(xiàn)的

ThreadLocal.get()

我們從get方法來分析,可以看到方法中獲取當(dāng)前線程,并通過當(dāng)前線程得到一個 ThreadLocalMap姥饰,我們可以暫時把這個ThreadLocalMap 理解為我們熟悉的HashMap列粪,然后通過 this(當(dāng)前ThreadLocal對象)作為key岂座,從Map中獲取Entry

圖1

我們再來看下,ThreadLocalMap 以及ThreadLocalMap.Entry 中的核心成員變量钾恢,ThreadLocalMap 中實現(xiàn)了一個簡單的hash表

圖2

看到這里你可能還不是很清晰泉懦,結(jié)合下面這張圖理解一下,每個線程(Thread對象)中有一個ThreadLocalMap崩哩,使線程之間的數(shù)據(jù)天然隔離邓嘹,ThreadLocalMap 有一張hash表 Entry[]汹押,每個 Entry 中對應(yīng)存儲著一個ThreadLocal實例 - value鸯乃,這樣使得不同的ThreadLocal 對象之間也形成了隔離

圖片來自網(wǎng)絡(luò)
ThreadLocalMap 中的hash表

我們通過 ThreadLocalMap.set() 來了解下內(nèi)部的hash表是如何實現(xiàn)的

圖3

線性探測是指當(dāng)發(fā)生hash沖突時鸟悴,利用固定的算法尋找一定步長的下個位置(ThreadLocal中發(fā)生hash沖突時奖年,index+1),依次判斷震贵,直至找到能夠存放的位置

如果線程中操作了大量的 ThreadLocal 對象猩系,勢必會造成hash沖突中燥,這是沒有必要的性能開銷,如果可以的話疗涉,我們可以只保留一個ThreadLocal對象

關(guān)于 ThreadLocal 的一些思考

  1. 為什么要使用弱引用

圖3中咱扣,我們看到hash表中會出現(xiàn) key == null的Entry,這是因為 ThreadLocalMap.Entry 的key (Entry 對ThreadLocal設(shè)置了弱引用沪铭,可以回顧一下圖2)

弱引用的對象擁有更短暫的生命周期。在GC時火窒,一旦發(fā)現(xiàn)了對象只具有弱引用熏矿,這個對象一定被回收

這么做的原因:如果ThreadLocal 對象需要被回收時(此時并沒有調(diào)用ThreadLocal.remove)离钝,線程中的ThreadLocalMap 一直強引用著 ThreadLocal對象,這會讓 ThreadLocal對象 以及對應(yīng)的value對象內(nèi)存無法釋放慧域,導(dǎo)致內(nèi)存泄漏昔榴。這算是ThreadLocal的一種容錯機制碘橘,這樣做使得了ThreadLocal對象得到了回收痘拆,但是value的內(nèi)存并沒有釋放纺蛆,所以ThreadLocalMap 的get、set方法中都會去嘗試清理ThreadLocal已經(jīng)被回收的entry温峭。

  1. 使用過后不及時remove會怎么樣

很多博客中都強調(diào)了凤藏,ThreadLocal.remove的重要性清笨。舉個例子刃跛,我們新啟了一個線程在這個線程中使用了ThreadLocal苛萎,我們并沒有調(diào)用remove,這會導(dǎo)致存儲的value對象一直沒有辦法被回收蛙酪,直到線程被銷毀

  1. 線程池中也需要remove嗎

以web線程池為例,如果每次都在過濾器中操作同一個ThreadLocal.set凹蜂,然后業(yè)務(wù)代碼中g(shù)et玛痊,似乎沒什么問題狂打。計算出的hash值都是一樣的,槽位也是一樣的會覆蓋上一次的值对省。確實業(yè)務(wù)不會有問題蒿涎,但是還是推薦大家在使用完之后remove同仆,因為這樣會讓無用的value對象早點被回收俗批,在很多java源碼中都會看到岁忘,對一些不再使用的對象進(jìn)行如下的help GC操作

object = null // help GC

所以我們也需要讓無用的對象失去引用干像,幫助GC

  1. 綜上所述

ThreadLocal 使用過后要及時remove麻汰,幫助JVM釋放內(nèi)存

參考

http://www.reibang.com/p/98b68c97df9b

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末五鲫,一起剝皮案震驚了整個濱河市位喂,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌七冲,老刑警劉巖澜躺,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異通铲,居然都是意外死亡颅夺,警方通過查閱死者的電腦和手機蛹稍,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進(jìn)店門拗慨,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人奉芦,你說我怎么就攤上這事赵抢。” “怎么了声功?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長先巴。 經(jīng)常有香客問我其爵,道長,這世上最難降的妖魔是什么伸蚯? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任摩渺,我火速辦了婚禮,結(jié)果婚禮上剂邮,老公的妹妹穿的比我還像新娘摇幻。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布囚企。 她就那樣靜靜地躺著伤疙,像睡著了一般黍特。 火紅的嫁衣襯著肌膚如雪灭衷。 梳的紋絲不亂的頭發(fā)上瞳遍,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天注祖,我揣著相機與錄音婚夫,去河邊找鬼靴庆。 笑死奢讨,一個胖子當(dāng)著我的面吹牛扒袖,可吹牛的內(nèi)容都是我干的飒泻。 我是一名探鬼主播,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼佩伤,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了纵潦?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤涛目,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體垮兑,經(jīng)...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年嚎卫,在試婚紗的時候發(fā)現(xiàn)自己被綠了麻昼。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,161評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡季春,死狀恐怖撵颊,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤词身,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布路星,位于F島的核電站,受9級特大地震影響友绝,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜切威,卻給世界環(huán)境...
    茶點故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一刺彩、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧叹螟,春花似錦象踊、人聲如沸袖外。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春紊馏,著一層夾襖步出監(jiān)牢的瞬間赫编,已是汗流浹背奋隶。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工萍聊, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人此衅。 一個月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親羡宙。 傳聞我的和親對象是個殘疾皇子探颈,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,916評論 2 344

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