ThreadLocal源碼解讀

一.threadlocal概述

顧名思義線(xiàn)程本地存儲(chǔ)阀蒂,如果定義了一個(gè)threadlocal對(duì)象该窗,每個(gè)線(xiàn)程往這個(gè)threadlocal對(duì)象中的讀寫(xiě)是隔離的,可以為相同對(duì)象在不同線(xiàn)程中都創(chuàng)建不同的存儲(chǔ)蚤霞,因此如果多個(gè)線(xiàn)程都使用變量x酗失,那線(xiàn)程本地存儲(chǔ)會(huì)生成多個(gè)用于x的不同存儲(chǔ)塊。

二.threadlocal實(shí)現(xiàn)的大致思路

Thread類(lèi)有一個(gè)ThreadLocal.threadLocalMap類(lèi)型的實(shí)例變量threadlocals争便,也就是說(shuō)每個(gè)線(xiàn)程都有一個(gè)threadLocalMap级零,threadLocalMap為定義在threadlocal內(nèi)的靜態(tài)內(nèi)部類(lèi),雖是一個(gè)map但沒(méi)有實(shí)現(xiàn)map接口滞乙,有自己獨(dú)立的實(shí)現(xiàn)奏纪。threadLocalMap的是使用開(kāi)放尋址線(xiàn)性探測(cè)的哈希表,節(jié)點(diǎn)Entry繼承自弱引用WeakReference斩启,本身為指向threadlocal的弱引用序调,有一個(gè)value屬性用來(lái)存放放到threadlocal內(nèi)的值⊥么兀可以簡(jiǎn)單的認(rèn)為key為threadlocal對(duì)象发绢,值為放入threadlocal的具體值。每個(gè)線(xiàn)程往threadlocal中塞值的時(shí)候垄琐,都會(huì)以該threadlocal對(duì)象為key边酒,往自己的threadlocalmap中存,讀也是以某個(gè)threadlocal為引用狸窘,在自己的threadlocalmap中找對(duì)應(yīng)的key墩朦,從而實(shí)現(xiàn)線(xiàn)程隔離。
圖為T(mén)hread類(lèi)中threadlocalmap實(shí)例變量:

ThreadLocal.ThreadLocalMap threadLocals = null;

三.API

image.png

threadLocalMap的api:


image.png

四.threadlocal API

get方法:

    public T get() {
        //獲取當(dāng)前線(xiàn)程的threadlocalmap
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        //如果線(xiàn)程中的threadlocalmap非空翻擒,則以這個(gè)threadlocal為key查詢(xún)對(duì)應(yīng)的value
        if (map != null) {
            ThreadLocalMap.Entry e = map.getEntry(this);
            if (e != null) {
                @SuppressWarnings("unchecked")
                T result = (T)e.value;
                return result;
            }
        }
        //否則設(shè)置初始化值
        return setInitialValue();
    }

    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;
    }
    //可以客戶(hù)端自己繼承threadlocal重寫(xiě)該類(lèi)氓涣,否則如果沒(méi)有set就get的話(huà)會(huì)拋出空指針錯(cuò)誤
    protected T initialValue() {
        return null;
    }

set,remove:

    public void set(T value) {
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null)
            map.set(this, value);
        else
            createMap(t, value);
    }

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

可以看出get set remove方法都是獲得當(dāng)前線(xiàn)程的threadlocalmap,然后調(diào)用他的get set remove方法陋气。

五.ThreadLocalMap詳解

entry:

        static class Entry extends WeakReference<ThreadLocal<?>> {
            /** The value associated with this ThreadLocal. */
            Object value;
            Entry(ThreadLocal<?> k, Object v) {
                super(k);
                value = v;
            }
        }

entry本身為一個(gè)弱引用劳吠,定義了一個(gè)類(lèi)型為object的value

1.1為什么使用弱引用:
如果使用普通的key-value形式存儲(chǔ),那么就造成threadlocal對(duì)象的生命周期與線(xiàn)程一樣長(zhǎng)巩趁,因?yàn)榭倳?huì)存在thread->threadlocalmap->entry->threadlocal這樣一條引用鏈存在痒玩,只要線(xiàn)程沒(méi)被銷(xiāo)毀,只要線(xiàn)程沒(méi)有被銷(xiāo)毀议慰,那么節(jié)點(diǎn)在gc可達(dá)性分析中一直處于可達(dá)狀態(tài)凰荚,無(wú)法被gc回收。而使用弱引用褒脯,如果threadlocal沒(méi)有強(qiáng)引用可達(dá)便瑟,那么它不會(huì)活過(guò)下次gc,而該threadlocal對(duì)應(yīng)的entry會(huì)失效番川,這位threadlocalmap垃圾清理提供了便利到涂。

1.2類(lèi)的成員變量和相應(yīng)方法:

/**
         * 初始容量16脊框,容量必須為2的冪
         */
        private static final int INITIAL_CAPACITY = 16;

        /**
         * 內(nèi)部其實(shí)是entry類(lèi)型的數(shù)組
         */
        private Entry[] table;

        /**
         * 表中entry的個(gè)數(shù)
         */
        private int size = 0;

        /**
         * 裝載因子
         */
        private int threshold; // Default to 0

        /**
         * 最差的裝載因子為2/3
         */
        private void setThreshold(int len) {
            threshold = len * 2 / 3;
        }

threadlocalmap為使用線(xiàn)性探測(cè)的哈希表,nextIndex和prevIndex為獲取表前一個(gè)元素和后一個(gè)元素的方法:

        /**
         * Increment i modulo len.
         */
        private static int nextIndex(int i, int len) {
            return ((i + 1 < len) ? i + 1 : 0);
        }

        /**
         * Decrement i modulo len.
         */
        private static int prevIndex(int i, int len) {
            return ((i - 1 >= 0) ? i - 1 : len - 1);
        }

由此可見(jiàn)threadlocalmap邏輯上來(lái)說(shuō)為一個(gè)環(huán)形践啄。

set方法:

 private void set(ThreadLocal<?> key, Object value) {

            Entry[] tab = table;
            int len = tab.length;
            //key所在的槽位浇雹,使用該key的哈希值與上tab表的長(zhǎng)度減一,相當(dāng)于用len對(duì)threadLocalHashCode取模
            int i = key.threadLocalHashCode & (len-1);
            //若槽不為空屿讽,線(xiàn)性探測(cè)
            for (Entry e = tab[i];
                 e != null;
                 e = tab[i = nextIndex(i, len)]) {
                ThreadLocal<?> k = e.get();
                //如果槽中entry對(duì)應(yīng)的threadlocal對(duì)象與參數(shù)所傳入對(duì)象相同昭灵,則直接替換
                if (k == key) {
                    e.value = value;
                    return;
                }
                //若槽中entry弱引用指向的threadlocal對(duì)象已經(jīng)被gc回收,則該entry失效伐谈,替換
                if (k == null) {
                    replaceStaleEntry(key, value, i);
                    return;
                }
            }
            //否則new一個(gè)entry
            tab[i] = new Entry(key, value);
            int sz = ++size;
            if (!cleanSomeSlots(i, sz) && sz >= threshold)
                rehash();
        }

未完待續(xù)

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末烂完,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子诵棵,更是在濱河造成了極大的恐慌抠蚣,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,204評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件履澳,死亡現(xiàn)場(chǎng)離奇詭異嘶窄,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)距贷,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,091評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)柄冲,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人忠蝗,你說(shuō)我怎么就攤上這事羊初。” “怎么了什湘?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,548評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)晦攒。 經(jīng)常有香客問(wèn)我闽撤,道長(zhǎng),這世上最難降的妖魔是什么脯颜? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,657評(píng)論 1 293
  • 正文 為了忘掉前任哟旗,我火速辦了婚禮,結(jié)果婚禮上栋操,老公的妹妹穿的比我還像新娘闸餐。我一直安慰自己,他們只是感情好矾芙,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,689評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布舍沙。 她就那樣靜靜地躺著,像睡著了一般剔宪。 火紅的嫁衣襯著肌膚如雪拂铡。 梳的紋絲不亂的頭發(fā)上壹无,一...
    開(kāi)封第一講書(shū)人閱讀 51,554評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音感帅,去河邊找鬼斗锭。 笑死,一個(gè)胖子當(dāng)著我的面吹牛失球,可吹牛的內(nèi)容都是我干的岖是。 我是一名探鬼主播,決...
    沈念sama閱讀 40,302評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼实苞,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼豺撑!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起硬梁,我...
    開(kāi)封第一講書(shū)人閱讀 39,216評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤前硫,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后荧止,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體屹电,經(jīng)...
    沈念sama閱讀 45,661評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,851評(píng)論 3 336
  • 正文 我和宋清朗相戀三年跃巡,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了危号。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,977評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡素邪,死狀恐怖外莲,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情兔朦,我是刑警寧澤偷线,帶...
    沈念sama閱讀 35,697評(píng)論 5 347
  • 正文 年R本政府宣布,位于F島的核電站沽甥,受9級(jí)特大地震影響声邦,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜摆舟,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,306評(píng)論 3 330
  • 文/蒙蒙 一亥曹、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧恨诱,春花似錦媳瞪、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,898評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至厕鹃,卻和暖如春龙巨,著一層夾襖步出監(jiān)牢的瞬間笼呆,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,019評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工旨别, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留诗赌,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,138評(píng)論 3 370
  • 正文 我出身青樓秸弛,卻偏偏與公主長(zhǎng)得像铭若,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子递览,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,927評(píng)論 2 355

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

  • 1. 背景 ThreadLocal源碼解讀叼屠,網(wǎng)上面早已經(jīng)泛濫了,大多比較淺绞铃,甚至有的連基本原理都說(shuō)的很有問(wèn)題,包括...
    小陳阿飛閱讀 1,355評(píng)論 2 56
  • 前言 ThreadLocal很多同學(xué)都搞不懂是什么東西镜雨,可以用來(lái)干嘛。但面試時(shí)卻又經(jīng)常問(wèn)到儿捧,所以這次我和大家一起學(xué)...
    liangzzz閱讀 12,452評(píng)論 14 228
  • 前言 ThreadLocal很多同學(xué)都搞不懂是什么東西荚坞,可以用來(lái)干嘛。但面試時(shí)卻又經(jīng)常問(wèn)到菲盾,所以這次我和大家一起學(xué)...
    懶癌正患者閱讀 1,123評(píng)論 1 34
  • 一颓影、使用姿勢(shì) 二、數(shù)據(jù)結(jié)構(gòu) 三懒鉴、源碼分析 四诡挂、回收機(jī)制 總結(jié) 一、使用姿勢(shì) 最佳實(shí)踐 在類(lèi)中定義ThreadLoc...
    原水寒閱讀 1,593評(píng)論 2 8
  • 七絕·桃花慵懶江南處處桃临谱,灼灼其華媚而嬌璃俗,無(wú)邊春色隨風(fēng)去,向北偷偷說(shuō)寂寥悉默。
    溪竹青閱讀 430評(píng)論 2 11