ThreadLocal的適用場景

摘抄自https://www.cnblogs.com/coderxx/p/12043764.html

ThreadLoal 變量,它的基本原理是漩绵,同一個 ThreadLocal 所包含的對象(對ThreadLocal< StringBuilder >而言即為 StringBuilder 類型變量)旨巷,在不同的 Thread 中有不同的副本(實際上是不同的實例):

因為每個 Thread 內(nèi)有自己的實例副本,且該副本只能由當前 Thread 使用岔留;
既然其它 Thread 不可訪問胁赢,那就不存在多線程間共享的問題务唐。

ThreadLocal 提供了線程本地的實例。它與普通變量的區(qū)別在于又兵,每個使用該變量的線程都會初始化一個完全獨立的實例副本任柜。ThreadLocal 變量通常被private static修飾。當一個線程結(jié)束時沛厨,它所使用的所有 ThreadLocal 相對的實例副本都會被回收宙地。

因此ThreadLocal 非常適用于這樣的場景:每個線程需要自己獨立的實例且該實例需要在多個方法中使用

ThreadLocalMap與內(nèi)存泄漏

在該方案中俄烁,Map 由 ThreadLocal 類的靜態(tài)內(nèi)部類 ThreadLocalMap 提供绸栅。該類的實例維護某個 ThreadLocal 與具體實例的映射。與 HashMap 不同的是页屠,ThreadLocalMap 的每個 Entry 都是一個對 Key 的弱引用粹胯,這一點我們可以從super(k)可看出。另外辰企,每個 Entry 中都包含了一個對 Value 的強引用风纠。

static class Entry extends WeakReference<ThreadLocal<?>> {
  /** The value associated with this ThreadLocal. */
  Object value;

  Entry(ThreadLocal<?> k, Object v) {
    super(k);
    value = v;
  }
}

之所以使用弱引用,是因為當沒有強引用指向 ThreadLocal 變量時牢贸,這個變量就可以被回收竹观,就避免ThreadLocal 因為不能被回收而造成的內(nèi)存泄漏的問題。

但是潜索,這里又可能出現(xiàn)另外一種內(nèi)存泄漏的問題臭增。ThreadLocalMap 維護 (key)ThreadLocal 變量與(value)具體實例的映射,當 ThreadLocal 變量被回收后竹习,該映射的鍵變?yōu)?null誊抛,該 Entry 無法被移除。從而使得實例被該 Entry 引用而無法被回收造成內(nèi)存泄漏整陌。

注意:Entry是對 ThreadLocal 類型的弱引用拗窃,并不是具體實例的弱引用瞎领,因此還存在具體實例相關的內(nèi)存泄漏的問題。

防止內(nèi)存泄漏

對于已經(jīng)不再被使用且已被回收的 ThreadLocal 對象随夸,它在每個線程內(nèi)對應的實例由于被線程的 ThreadLocalMap 的 Entry 強引用九默,無法被回收,可能會造成內(nèi)存泄漏宾毒。

針對該問題驼修,ThreadLocalMap 的 set 方法中,通過 replaceStaleEntry 方法將所有鍵為 null 的 Entry 的值設置為 null诈铛,從而使得該值可被回收邪锌。另外,會在 rehash 方法中通過 expungeStaleEntry 方法將鍵和值為 null 的 Entry 設置為 null 從而使得該 Entry 可被回收癌瘾。

總結(jié)

ThreadLocal 并不解決線程間共享數(shù)據(jù)的問題
ThreadLocal 通過隱式的在不同線程內(nèi)創(chuàng)建獨立實例副本避免了實例線程安全的問題
每個線程持有一個 Map 并維護了 ThreadLocal 對象與具體實例的映射,該 Map 由于只被持有它的線程訪問饵溅,故不存在線程安全以及鎖的問題
ThreadLocalMap 的 Entry 對 ThreadLocal 的引用為弱引用妨退,避免了 ThreadLocal 對象無法被回收的問題
ThreadLocalMap 的 set 方法通過調(diào)用 replaceStaleEntry 方法回收鍵為 null 的 Entry 對象的值(即為具體實例)以及 Entry 對象本身從而防止內(nèi)存泄漏
ThreadLocal 適用于變量在線程間隔離且在方法間共享的場景

?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市蜕企,隨后出現(xiàn)的幾起案子咬荷,更是在濱河造成了極大的恐慌,老刑警劉巖轻掩,帶你破解...
    沈念sama閱讀 218,682評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件幸乒,死亡現(xiàn)場離奇詭異,居然都是意外死亡唇牧,警方通過查閱死者的電腦和手機罕扎,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,277評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來丐重,“玉大人腔召,你說我怎么就攤上這事“绲耄” “怎么了臀蛛?”我有些...
    開封第一講書人閱讀 165,083評論 0 355
  • 文/不壞的土叔 我叫張陵,是天一觀的道長崖蜜。 經(jīng)常有香客問我浊仆,道長,這世上最難降的妖魔是什么豫领? 我笑而不...
    開封第一講書人閱讀 58,763評論 1 295
  • 正文 為了忘掉前任抡柿,我火速辦了婚禮,結(jié)果婚禮上氏堤,老公的妹妹穿的比我還像新娘沙绝。我一直安慰自己搏明,他們只是感情好,可當我...
    茶點故事閱讀 67,785評論 6 392
  • 文/花漫 我一把揭開白布闪檬。 她就那樣靜靜地躺著星著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪粗悯。 梳的紋絲不亂的頭發(fā)上虚循,一...
    開封第一講書人閱讀 51,624評論 1 305
  • 那天,我揣著相機與錄音样傍,去河邊找鬼横缔。 笑死,一個胖子當著我的面吹牛衫哥,可吹牛的內(nèi)容都是我干的茎刚。 我是一名探鬼主播,決...
    沈念sama閱讀 40,358評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼撤逢,長吁一口氣:“原來是場噩夢啊……” “哼膛锭!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起蚊荣,我...
    開封第一講書人閱讀 39,261評論 0 276
  • 序言:老撾萬榮一對情侶失蹤初狰,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后互例,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體奢入,經(jīng)...
    沈念sama閱讀 45,722評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,900評論 3 336
  • 正文 我和宋清朗相戀三年媳叨,在試婚紗的時候發(fā)現(xiàn)自己被綠了腥光。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,030評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡糊秆,死狀恐怖柴我,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情扩然,我是刑警寧澤艘儒,帶...
    沈念sama閱讀 35,737評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站夫偶,受9級特大地震影響界睁,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜兵拢,卻給世界環(huán)境...
    茶點故事閱讀 41,360評論 3 330
  • 文/蒙蒙 一翻斟、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧说铃,春花似錦访惜、人聲如沸嘹履。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,941評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽砾嫉。三九已至,卻和暖如春窒篱,著一層夾襖步出監(jiān)牢的瞬間焕刮,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,057評論 1 270
  • 我被黑心中介騙來泰國打工墙杯, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留配并,地道東北人。 一個月前我還...
    沈念sama閱讀 48,237評論 3 371
  • 正文 我出身青樓高镐,卻偏偏與公主長得像溉旋,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子嫉髓,可洞房花燭夜當晚...
    茶點故事閱讀 44,976評論 2 355