ThreadLocal原理

1.簡介

在使用多線程場景中嘲恍,對(duì)同一個(gè)共享變量進(jìn)行訪問息堂,會(huì)容易出現(xiàn)并發(fā)問題嚷狞,造成所得到的結(jié)果不正確,通常采取的辦法是在訪問這個(gè)共享變量進(jìn)行同步操作荣堰,一般加鎖操作床未。

但是,眾所周知振坚,加鎖操作薇搁,一定程度上會(huì)減低程序性能。假如這個(gè)變量并不需要保證在各個(gè)線程共享渡八,各個(gè)線程都可以獨(dú)立操作這個(gè)變量時(shí)啃洋,不用在意是否在讀取這個(gè)變量時(shí)候,這個(gè)變量已被其他線程修改了值屎鳍,那么完全可以不用同步操作宏娄,這就涉及到我們今天要講的 ThreadLocal 這個(gè)類。

ThreaLocal 是 JKD 包提供逮壁,位于 java.lang 包下孵坚,它提供線程本地變量,簡而言之,就是為每一個(gè)線程創(chuàng)建一個(gè)變量副本卖宠,線程操作是自己本地內(nèi)存中的值巍杈,從而使每個(gè)線程都可以對(duì)這個(gè)變量為所欲為了。典型的應(yīng)用場景是逗堵,數(shù)據(jù)庫打開關(guān)閉操作秉氧。

2.使用

ThreadLocal 使用也非常簡單,下面寫一個(gè)簡單例子:

@Test
public void testThreadLocal() {
    final ThreadLocal<String> local = new ThreadLocal<>();
    Thread threadA = new Thread(new Runnable() {
         @Override
         public void run() {
             local.set("線程A對(duì)變量為所欲為");
             System.out.println("線程A:" + local.get());
         }
    });
    Thread threadB = new Thread(new Runnable() {
         @Override
         public void run() {
             local.set("線程B對(duì)變量為所欲為");
             System.out.println("線程B:" + local.get());
         }
    });
    threadA.start();
    threadB.start();
}
//輸出:
//線程A:線程A對(duì)變量為所欲為
//線程B:線程B對(duì)變量為所欲為

以上程序可以看出蜒秤,兩個(gè)線程都通過 set 對(duì) local 設(shè)置值汁咏,之后通過 get 讀取 local 值,但是都是操作自己本地副本作媚。

3.原理

ThreaLocal 的代碼量很少攘滩,原理也比較簡單,簡單來說纸泡,內(nèi)部維護(hù)了一個(gè) Map漂问,這個(gè) Map 是 ThreadLocal 一個(gè)內(nèi)部靜態(tài)類,叫 ThreadLocalMap女揭。因?yàn)槊恳粋€(gè)線程都可以關(guān)聯(lián)多個(gè) ThreadLocal 變量蚤假,所以設(shè)計(jì)成 Map結(jié)構(gòu)。接下來看看各個(gè)方法主要實(shí)現(xiàn)邏輯:

3.1.void set(T value)

public void set(T value) {
     //1.獲取當(dāng)前線程
     Thread t = Thread.currentThread();
     //2.以當(dāng)前線程 t 為 key吧兔,獲取當(dāng)前 threadLocals 變量
     ThreadLocalMap map = getMap(t);
     //3.如果存在磷仰,把 value 設(shè)置到 threadLocals 中
     if (map != null)
         map.set(this, value);
     //4.如果不存在,創(chuàng)建當(dāng)前線程對(duì)應(yīng)的 Map
     else
         createMap(t, value);
}

ThreadLocalMap getMap(Thread t) {
    //獲取線程自己的變量 threadLocals
    return t.threadLocals;
}

void createMap(Thread t, T firstValue) {
     //創(chuàng)建當(dāng)前線程的 threadLocals 變量
     t.threadLocals = new ThreadLocalMap(this, firstValue);
}

3.2.T get(T value)

public T get() {
    //1.獲取當(dāng)前線程
     Thread t = Thread.currentThread();
    //2.以當(dāng)前線程 t 為 key境蔼,獲取當(dāng)前 threadLocals 變量
     ThreadLocalMap map = getMap(t);
    //3.如果不為 null灶平,就返回 key 對(duì)應(yīng)的本地變量的值
     if (map != null) {
         ThreadLocalMap.Entry e = map.getEntry(this);
         if (e != null) {
             @SuppressWarnings("unchecked")
             T result = (T)e.value;
             return result;
         }
     }
    //4.如果為null,就進(jìn)行初始化
     return setInitialValue();
}

//邏輯和 void set() 差不多
private T setInitialValue() {
      T value = initialValue();
      Thread t = Thread.currentThread();
      ThreadLocalMap map = getMap(t);
      if (map != null)
          map.set(this, value);
      else
          createMap(t, value);
      return value;
}

protected T initialValue() {
      return null;
}

3.3.void remove()

public void remove() {
     //1.以當(dāng)前線程 t 為 key箍土,獲取當(dāng)前 threadLocals 變量
     ThreadLocalMap m = getMap(Thread.currentThread());
     //2.如果不為 null逢享,刪除本地變量
     if (m != null)
         m.remove(this);
}

4.問題

在使用 ThreadLocal 中,由于每一個(gè)線程都持有一個(gè) threadLocals 變量吴藻,他是一個(gè)HashMap類型瞒爬,key 為當(dāng)前線程,value 為變量副本沟堡,如果一個(gè)線程一直不消亡疮鲫,會(huì)一直持有變量副本,容易造成內(nèi)存泄露弦叶,所以俊犯,為了保險(xiǎn)起見,在使用完變量副本之后伤哺,要調(diào)用 remove 方法來刪除本地變量副本燕侠。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末者祖,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子绢彤,更是在濱河造成了極大的恐慌七问,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,104評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件茫舶,死亡現(xiàn)場離奇詭異械巡,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)饶氏,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,816評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門讥耗,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人疹启,你說我怎么就攤上這事古程。” “怎么了喊崖?”我有些...
    開封第一講書人閱讀 168,697評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵挣磨,是天一觀的道長。 經(jīng)常有香客問我荤懂,道長茁裙,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,836評(píng)論 1 298
  • 正文 為了忘掉前任节仿,我火速辦了婚禮晤锥,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘粟耻。我一直安慰自己查近,他們只是感情好眉踱,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,851評(píng)論 6 397
  • 文/花漫 我一把揭開白布挤忙。 她就那樣靜靜地躺著,像睡著了一般谈喳。 火紅的嫁衣襯著肌膚如雪册烈。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,441評(píng)論 1 310
  • 那天婿禽,我揣著相機(jī)與錄音赏僧,去河邊找鬼。 笑死扭倾,一個(gè)胖子當(dāng)著我的面吹牛淀零,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播膛壹,決...
    沈念sama閱讀 40,992評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼驾中,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼唉堪!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起肩民,我...
    開封第一講書人閱讀 39,899評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤唠亚,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后持痰,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體灶搜,經(jīng)...
    沈念sama閱讀 46,457評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,529評(píng)論 3 341
  • 正文 我和宋清朗相戀三年工窍,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了割卖。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,664評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡移剪,死狀恐怖究珊,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情纵苛,我是刑警寧澤剿涮,帶...
    沈念sama閱讀 36,346評(píng)論 5 350
  • 正文 年R本政府宣布,位于F島的核電站攻人,受9級(jí)特大地震影響取试,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜怀吻,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,025評(píng)論 3 334
  • 文/蒙蒙 一瞬浓、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧蓬坡,春花似錦猿棉、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,511評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至兆龙,卻和暖如春杖爽,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背紫皇。 一陣腳步聲響...
    開封第一講書人閱讀 33,611評(píng)論 1 272
  • 我被黑心中介騙來泰國打工慰安, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人聪铺。 一個(gè)月前我還...
    沈念sama閱讀 49,081評(píng)論 3 377
  • 正文 我出身青樓化焕,卻偏偏與公主長得像,于是被迫代替她去往敵國和親铃剔。 傳聞我的和親對(duì)象是個(gè)殘疾皇子撒桨,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,675評(píng)論 2 359

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

  • 1. ThreadLocal的簡介 在多線程編程中通常解決線程安全的問題我們會(huì)利用synchronzed或者loc...
    先生zeng閱讀 705評(píng)論 2 8
  • 原文鏈接:Java進(jìn)階(七)正確理解Thread Local的原理與適用場景 - 郭俊Jason - 博客園 Th...
    Walter_Hu閱讀 771評(píng)論 0 1
  • 總述 ThreadLocal 在面試中經(jīng)常提到脂倦,關(guān)于ThreadLocal使用不當(dāng)造成OOM以及在特殊場景下,通過...
    墨染書閱讀 16,076評(píng)論 11 44
  • ThreadLocal作用 對(duì)于Android程序員來說元莫,很多人都是在學(xué)習(xí)消息機(jī)制時(shí)候了解到ThreadLocal...
    三雒閱讀 1,321評(píng)論 0 19
  • 電影《春嬌與志明》有句廣為流傳的臺(tái)詞:“一輩子那么長赖阻,誰沒愛上過幾個(gè)人渣”。這句話放之四海而皆準(zhǔn)踱蠢,但是阿嬌卻不認(rèn)同...
    麥一十閱讀 513評(píng)論 7 10