ThreadLocal解析

原理

產(chǎn)生線程安全問題的根源在于多線程之間的數(shù)據(jù)共享。如果沒有數(shù)據(jù)共享,就沒有多線程并發(fā)安全問題。ThreadLocal就是用來避免多線程數(shù)據(jù)共享從而避免多線程并發(fā)安全問題。它為每個(gè)線程保留一個(gè)對(duì)象的副本,避免了多線程數(shù)據(jù)共享。每個(gè)線程作用的對(duì)象都是線程私有的一個(gè)對(duì)象拷貝。一個(gè)線程的對(duì)象副本無法被其他線程訪問到(InheritableThreadLocal除外)。

注意ThreadLocal并不是一種多線程并發(fā)安全問題的解決方案,因?yàn)門hreadLocal的原理在于避免多線程數(shù)據(jù)共享從而實(shí)現(xiàn)線程安全。

來看一下JDK文檔中ThreadLocal的描述:

This class provides thread-local variables.? These variables differ fromtheir normal counterpartsinthat each thread that accesses one (via its{@code get} or {@codeset} method) has its own, independently initializedcopy of the variable.? {@code ThreadLocal} instances are typically privatestatic fieldsinclasses that wish to associate state with a thread (e.g.,a user ID or Transaction ID).Each thread holds an implicit reference to its copy of a thread-localvariable as long as the thread is alive and the {@code ThreadLocal}instance is accessible; after a thread goes away, all of its copies ofthread-local instances are subject to garbage collection (unless otherreferences to these copies exist).

其大致的意思是:ThreadLocal為每個(gè)線程保留一個(gè)對(duì)象的副本,通過set()方法和get()方法來設(shè)置和獲取對(duì)象。ThreadLocal通常被定義為私有的和靜態(tài)的,用于關(guān)聯(lián)線程的某些狀態(tài)。當(dāng)關(guān)聯(lián)ThreadLocal的線程死亡后,ThreadLocal實(shí)例才可以被GC。也就是說如果ThreadLocal關(guān)聯(lián)的線程如果沒有死亡,則ThreadLocal就一直不能被回收。

用法

創(chuàng)建

ThreadLocal<String> mThreadLocal = new ThreadLocal<>();

set方法

mThreadLocal.set(Thread.currentThread().getName());

get方法

String mThreadLocalVal = mThreadLocal.get();

設(shè)置初始值

ThreadLocal<String> mThreadLocal = ThreadLocal.withInitial(() -> Thread.currentThread().getName());

完整示例

import java.util.concurrent.atomic.AtomicInteger;public class ThreadId {? ? // Atomicintegercontaining the next thread ID to be assigned? ? private static final AtomicInteger nextId = new AtomicInteger(0);? ? // Threadlocalvariable containing each thread's ID

? ? private static final ThreadLocal<Integer> threadId =

? ? ? ? ? ? ThreadLocal.withInitial(() -> nextId.getAndIncrement());

? ? // Returns the current thread's unique ID, assigning itifnecessary? ? public static intget() {returnthreadId.get();? ? }}

源碼分析

ThreadLocal為每個(gè)線程維護(hù)一個(gè)哈希表,用于保存線程本地變量,哈希表的key是ThreadLocal實(shí)例,value就是需要保存的對(duì)象。

publlic class Thread {? ? ...? ? /* ThreadLocal values pertaining to this thread. This map is maintained? ? * by the ThreadLocal class. */? ? ThreadLocal.ThreadLocalMap threadLocals = null;? ? ...}public class ThreadLocal {? ? ...? ? static class ThreadLocalMap {? ? ? ? static class Entry extends WeakReference> {? ? ? ? ...? ? ...}

threadLocals是非靜態(tài)的,也就是說每個(gè)線程都會(huì)有一個(gè)ThreadLocalMap哈希表用來保存本地變量。ThreadLocalMap的Entry的鍵(ThreadLocal<?>)是弱引用的,也就是說當(dāng)垃圾收集器發(fā)現(xiàn)這個(gè)弱引用的鍵時(shí)不管內(nèi)存是否足夠多將其回收。這里回收的是ThreadLocalMap的Entry的ThreadLocal而不是Entry,因此還是可能會(huì)造成內(nèi)存泄露。

get()方法

public Tget() {? ? Thread t = Thread.currentThread();? ? ThreadLocalMap map = getMap(t); // 獲取線程關(guān)聯(lián)的ThreadLocalMap哈希表if(map != null) {? ? ? ? ThreadLocalMap.Entry e = map.getEntry(this); // 獲取entryif(e != null) {? ? ? ? ? ? @SuppressWarnings("unchecked")? ? ? ? ? ? T result = (T)e.value; // 返回entry關(guān)聯(lián)的對(duì)象returnresult;? ? ? ? }? ? }returnsetInitialValue(); // 如果當(dāng)前線程關(guān)聯(lián)的本地變量哈希表為空,則創(chuàng)建一個(gè)}ThreadLocalMap getMap(Thread t) {returnt.threadLocals;}

首先通過getMap()方法獲取當(dāng)前線程的ThreadLocalMap實(shí)例,ThreadLocalMap是線程私有的,因此這里是線程安全的。ThreadLocalMap的getEntry()方法用于獲取當(dāng)前線程關(guān)聯(lián)的ThreadLocalMap鍵值對(duì),如果鍵值對(duì)不為空則返回值。如果鍵值對(duì)為空,則通過setInitialValue()方法設(shè)置初始值,并返回。注意setInitialValue()方法是private,是不可以覆寫的。

設(shè)置初始值

private TsetInitialValue() {? ? T value = initialValue();? ? Thread t = Thread.currentThread();? ? ThreadLocalMap map = getMap(t);if(map != null)? ? ? ? map.set(this, value);elsecreateMap(t, value);returnvalue;}protected TinitialValue() {returnnull;}

設(shè)置初始值會(huì)調(diào)用initialValue()方法獲取初始值,該方法默認(rèn)返回null,該方法可以被覆寫,用于設(shè)置初始值。例如上面的例子中,通過匿名內(nèi)部類覆寫了initialValue()方法設(shè)置了初始值。獲取到初始值后,判斷當(dāng)前線程關(guān)聯(lián)的本地變量哈希表是否為空,如果非空則設(shè)置初始值,否則先新建本地變量哈希表再設(shè)置初始值。最后返回這個(gè)初始值。

set()方法

public voidset(T value) {? ? Thread t = Thread.currentThread();? ? ThreadLocalMap map = getMap(t);if(map != null)? ? ? ? map.set(this, value);elsecreateMap(t, value);}

該方法先獲取該線程的 ThreadLocalMap 對(duì)象,然后直接將 ThreadLocal 對(duì)象(即代碼中的 this)與目標(biāo)實(shí)例的映射添加進(jìn) ThreadLocalMap 中。當(dāng)然,如果映射已經(jīng)存在,就直接覆蓋。另外,如果獲取到的 ThreadLocalMap 為 null,則先創(chuàng)建該 ThreadLocalMap 對(duì)象。

防止內(nèi)存泄露

前面分析得知ThreadLocalMap的Entry的key是弱引用的,key可以在垃圾收集器工作的時(shí)候就被回收掉,但是存在 當(dāng)前線程->ThreadLocal->ThreadLocalMap->Entry的一條強(qiáng)引用鏈,因此如果當(dāng)前線程沒有死亡,或者還持有ThreadLocal實(shí)例的引用Entry就無法被回收。從而造成內(nèi)存泄露。

當(dāng)我們使用線程池來處理請(qǐng)求的時(shí)候,一個(gè)請(qǐng)求處理完成,線程并不一定會(huì)被回收,因此線程還會(huì)持有ThreadLocal實(shí)例的引用,即使ThreadLocal已經(jīng)沒有作用了。此時(shí)就發(fā)生了ThreadLocal的內(nèi)存泄露。

針對(duì)該問題,ThreadLocalMap 的 set 方法中,通過 replaceStaleEntry 方法將所有鍵為 null 的 Entry 的值設(shè)置為 null,從而使得該值可被回收。另外,會(huì)在 rehash 方法中通過 expungeStaleEntry 方法將鍵和值為 null 的 Entry 設(shè)置為 null 從而使得該 Entry 可被回收。通過這種方式,ThreadLocal 可防止內(nèi)存泄漏。

private voidset(ThreadLocal key, Object value) {? ? Entry[] tab = table;? ? int len = tab.length;? ? int i = key.threadLocalHashCode & (len-1);for(Entry e = tab[i]; e != null; e = tab[i = nextIndex(i, len)]) {? ? ? ? ThreadLocal k = e.get();if(k == key) {? ? ? ? ? ? e.value = value;return;? ? ? ? }if(k == null) {? ? ? ? ? ? replaceStaleEntry(key, value, i); // key為空,則代表該Entry不再需要,設(shè)置Entry的value指針和Entry指針為null,幫助GCreturn;? ? ? ? }? ? }? ? tab[i] = new Entry(key, value);? ? int sz = ++size;if(!cleanSomeSlots(i, sz) && sz >= threshold)rehash();}

使用場(chǎng)景

ThreadLocal場(chǎng)景的使用場(chǎng)景:

每個(gè)線程需要有自己?jiǎn)为?dú)的實(shí)例

實(shí)例需要在多個(gè)方法中共享,但不希望被多線程共享

例如用來解決數(shù)據(jù)庫連接、Session管理等。

InheritableThreadLocal

ThreadLocal為每個(gè)線程保留一個(gè)線程私有的對(duì)象副本,線程之間無法共享訪問,但是有一個(gè)例外:InheritableThreadLocal,InheritableThreadLocal可以實(shí)現(xiàn)在子線程中訪問父線程中的對(duì)象副本。下面是一個(gè)InheritableThreadLocal和ThreadLocal區(qū)別的例子:

public class InheritableThreadLocalExample {? ? public static void main(String[] args) throws InterruptedException {? ? ? ? InheritableThreadLocalintegerInheritableThreadLocal = new InheritableThreadLocal<>();? ? ? ? ThreadLocalintegerThreadLocal = new ThreadLocal<>();integerInheritableThreadLocal.set(1);integerThreadLocal.set(0);? ? ? ? Thread thread = new Thread(() -> System.out.println(Thread.currentThread().getName() +", "+integerThreadLocal.get() +" / "+integerInheritableThreadLocal.get()));? ? ? ? thread.start();? ? ? ? thread.join();? ? }

運(yùn)行上述代碼會(huì)發(fā)現(xiàn)在子線程中可以獲取父線程的InheritableThreadLocal中的變量,但是無法獲取父線程的ThreadLocal中的變量。

InheritableThreadLocal是ThreadLocal的子類:

public class InheritableThreadLocal extends ThreadLocal {? ? ...? ? // 覆寫了ThreadLocal的getMap方法,返回的是Thread中的inheritableThreadLocals? ? ThreadLocalMap getMap(Thread t) {returnt.inheritableThreadLocals;? ? }? ? // 覆寫了createMap方法,創(chuàng)建的也是Thread中的inheritableThreadLocals這個(gè)哈希表? ? void createMap(Thread t, T firstValue) {? ? ? ? t.inheritableThreadLocals = new ThreadLocalMap(this, firstValue);? ? }}

而Thread的inheritableThreadLocals會(huì)在線程初始化的時(shí)候進(jìn)行初始化。這個(gè)過程在Thread類的init()方法中:

private void init(ThreadGroup g, Runnable target, String name,? ? ? ? ? ? ? ? ? long stackSize, AccessControlContext acc) {? ? ...? ? Thread parent = currentThread();? ? ...if(parent.inheritableThreadLocals != null)? ? ? ? this.inheritableThreadLocals =? ? ? ? ? ? ThreadLocal.createInheritedMap(parent.inheritableThreadLocals);? ? ...}

一個(gè)線程在初始化的時(shí)候,會(huì)判斷創(chuàng)建這個(gè)線程的父線程的inheritableThreadLocals是否為空,如果不為空,則會(huì)拷貝父線程inheritableThreadLocals到當(dāng)前創(chuàng)建的子線程的inheritableThreadLocals中去。

當(dāng)我們?cè)谧泳€程調(diào)用get()方法時(shí),InheritableThreadLocal的getMap()方法返回的是Thread中的inheritableThreadLocals,而子線程的inheritableThreadLocals已經(jīng)拷貝了父線程的inheritableThreadLocals,因此在子線程中可以讀取父線程中的inheritableThreadLocals中保存的對(duì)象。

總結(jié)

ThreadLocal為每個(gè)線程保留對(duì)象副本,多線程之間沒有數(shù)據(jù)共享。因此它并不解決線程間共享數(shù)據(jù)的問題。

每個(gè)線程持有一個(gè)Map并維護(hù)了ThreadLocal對(duì)象與具體實(shí)例的映射,該Map由于只被持有它的線程訪問,故不存在線程安全以及鎖的問題。

ThreadLocalMap的Entry對(duì)ThreadLocal的引用為弱引用,避免了ThreadLocal對(duì)象無法被回收的問題。

ThreadLocalMap的set方法通過調(diào)用 replaceStaleEntry 方法回收鍵為 null的Entry 對(duì)象的值(即為具體實(shí)例)以及Entry對(duì)象本身從而防止內(nèi)存泄漏。

ThreadLocal 適用于變量在線程間隔離且在方法間共享的場(chǎng)景。

ThreadLocal中的變量是線程私有的,其他線程無法訪問到另外一個(gè)線程的變量。但是InheritableThreadLocal是個(gè)例外,通過InheritableThreadLocal可以在子線程中訪問到父線程中的變量。

在此我向大家推薦一個(gè)架構(gòu)學(xué)習(xí)交流群。交流學(xué)習(xí)群號(hào):938837867 暗號(hào):555 里面會(huì)分享一些資深架構(gòu)師錄制的視頻錄像:有Spring,MyBatis,Netty源碼分析,高并發(fā)、高性能、分布式、微服務(wù)架構(gòu)的原理,JVM性能優(yōu)化、分布式架構(gòu)等這些成為架構(gòu)師必備

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,363評(píng)論 6 532
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,497評(píng)論 3 416
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,305評(píng)論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,962評(píng)論 1 311
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,727評(píng)論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,193評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,257評(píng)論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,411評(píng)論 0 288
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,945評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,777評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,978評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,519評(píng)論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,216評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,642評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,878評(píng)論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,657評(píng)論 3 391
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,960評(píng)論 2 373

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

  • 原理 產(chǎn)生線程安全問題的根源在于多線程之間的數(shù)據(jù)共享。如果沒有數(shù)據(jù)共享,就沒有多線程并發(fā)安全問題。ThreadLo...
    zhong0316閱讀 379評(píng)論 0 4
  • 一、使用姿勢(shì) 二、數(shù)據(jù)結(jié)構(gòu) 三、源碼分析 四、回收機(jī)制 總結(jié) 一、使用姿勢(shì) 最佳實(shí)踐 在類中定義ThreadLoc...
    原水寒閱讀 1,624評(píng)論 2 8
  • 前言 在各大公司招聘筆試和面試題題中,都遇到了很多ThreadLocal的問題,最近博主在面試的時(shí)候也被兩次問到過...
    Kevin_ZGJ閱讀 437評(píng)論 1 3
  • 原文鏈接:Java進(jìn)階(七)正確理解Thread Local的原理與適用場(chǎng)景 - 郭俊Jason - 博客園 Th...
    Walter_Hu閱讀 776評(píng)論 0 1
  • 本篇文章的主要內(nèi)容如下: 1、Java中的ThreadLocal2、 Android中的ThreadLocal3、...
    Sophia_dd35閱讀 626評(píng)論 0 5