亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

解密ThreadLocal

2019-11-11 07:00:20
字體:
來源:轉載
供稿:網友

概述

相信讀者在網上也看了很多關于ThreadLocal的資料,很多博客都這樣說:ThreadLocal為解決多線程程序的并發問題提供了一種新的思路;ThreadLocal的目的是為了解決多線程訪問資源時的共享問題。如果你也這樣認為的,那現在給你10秒鐘,清空之前對ThreadLocal的錯誤的認知!看看JDK中的源碼是怎么寫的:

This class PRovides thread-local variables. These variables differ fromtheir normal counterparts in that each thread that accesses one (via its{@code get} or {@code set} method) has its own, independently initializedcopy of the variable. {@code ThreadLocal} instances are typically privatestatic fields in classes that wish to associate state with a thread (e.g.,a user ID or Transaction ID).

翻譯過來大概是這樣的(英文不好,如有更好的翻譯,請留言說明):

ThreadLocal類用來提供線程內部的局部變量。這種變量在多線程環境下訪問(通過get或set方法訪問)時能保證各個線程里的變量相對獨立于其他線程內的變量。ThreadLocal實例通常來說都是private static類型的,用于關聯線程和線程的上下文。

可以總結為一句話:ThreadLocal的作用是提供線程內的局部變量,這種變量在線程的生命周期內起作用,減少同一個線程內多個函數或者組件之間一些公共變量的傳遞的復雜度。舉個例子,我出門需要先坐公交再做地鐵,這里的坐公交和坐地鐵就好比是同一個線程內的兩個函數,我就是一個線程,我要完成這兩個函數都需要同一個東西:公交卡(北京公交和地鐵都使用公交卡),那么我為了不向這兩個函數都傳遞公交卡這個變量(相當于不是一直帶著公交卡上路),我可以這么做:將公交卡事先交給一個機構,當我需要刷卡的時候再向這個機構要公交卡(當然每次拿的都是同一張公交卡)。這樣就能達到只要是我(同一個線程)需要公交卡,何時何地都能向這個機構要的目的。

有人要說了:你可以將公交卡設置為全局變量啊,這樣不是也能何時何地都能取公交卡嗎?但是如果有很多個人(很多個線程)呢?大家可不能都使用同一張公交卡吧(我們假設公交卡是實名認證的),這樣不就亂套了嘛。現在明白了吧?這就是ThreadLocal設計的初衷:提供線程內部的局部變量,在本線程內隨時隨地可取,隔離其他線程。

ThreadLocal基本操作

構造函數

ThreadLocal的構造函數簽名是這樣的:

123456
/** * Creates a thread local variable. * @see #withInitial(java.util.function.Supplier) */public ThreadLocal() {}

內部啥也沒做。

initialValue函數

initialValue函數用來設置ThreadLocal的初始值,函數簽名如下:

123
protected T initialValue() {    return null;}

該函數在調用get函數的時候會第一次調用,但是如果一開始就調用了set函數,則該函數不會被調用。通常該函數只會被調用一次,除非手動調用了remove函數之后又調用get函數,這種情況下,get函數中還是會調用initialValue函數。該函數是protected類型的,很顯然是建議在子類重載該函數的,所以通常該函數都會以匿名內部類的形式被重載,以指定初始值,比如:

1234567891011121314
package com.winwill.test;/** * @author qifuguang * @date 15/9/2 00:05 */public class TestThreadLocal {    private static final ThreadLocal<Integer> value = new ThreadLocal<Integer>() {        @Override        protected Integer initialValue() {            return Integer.valueOf(1);        }    };}

get函數

該函數用來獲取與當前線程關聯的ThreadLocal的值,函數簽名如下:

1
public T get()

如果當前線程沒有該ThreadLocal的值,則調用initialValue函數獲取初始值返回。

set函數

set函數用來設置當前線程的該ThreadLocal的值,函數簽名如下:

1
public void set(T value)

設置當前線程的ThreadLocal的值為value。

remove函數

remove函數用來將當前線程的ThreadLocal綁定的值刪除,函數簽名如下:

1
public void remove()

在某些情況下需要手動調用該函數,防止內存泄露。

代碼演示

學習了最基本的操作之后,我們用一段代碼來演示ThreadLocal的用法,該例子實現下面這個場景:

有5個線程,這5個線程都有一個值value,初始值為0,線程運行時用一個循環往value值相加數字。

代碼實現:

123456789101112131415161718192021222324252627282930313233343536
package com.winwill.test;/** * @author qifuguang * @date 15/9/2 00:05 */public class TestThreadLocal {    private static final ThreadLocal<Integer> value = new ThreadLocal<Integer>() {        @Override        protected Integer initialValue() {            return 0;        }    };    public static void main(String[] args) {        for (int i = 0; i < 5; i++) {            new Thread(new MyThread(i)).start();        }    }    static class MyThread implements Runnable {        private int index;        public MyThread(int index) {            this.index = index;        }        public void run() {            System.out.println("線程" + index + "的初始value:" + value.get());            for (int i = 0; i < 10; i++) {                value.set(value.get() + i);            }            System.out.println("線程" + index + "的累加value:" + value.get());        }    }}

執行結果為:

線程0的初始value:0線程3的初始value:0線程2的初始value:0線程2的累加value:45線程1的初始value:0線程3的累加value:45線程0的累加value:45線程1的累加value:45線程4的初始value:0線程4的累加value:45

可以看到,各個線程的value值是相互獨立的,本線程的累加操作不會影響到其他線程的值,真正達到了線程內部隔離的效果。

如何實現的

看了基本介紹,也看了最簡單的效果演示之后,我們更應該好好研究下ThreadLocal內部的實現原理。如果給你設計,你會怎么設計?相信大部分人會有這樣的想法:

每個ThreadLocal類創建一個Map,然后用線程的ID作為Map的key,實例對象作為Map的value,這樣就能達到各個線程的值隔離的效果。

沒錯,這是最簡單的設計方案,JDK最早期的ThreadLocal就是這樣設計的。JDK1.3(不確定是否是1.3)之后ThreadLocal的設計換了一種方式。

我們先看看JDK8的ThreadLocal的get方法的源碼:

12345678910111213
public T get() {      Thread t = Thread.currentThread();      ThreadLocalMap map = getMap(t);      if (map != null) {          ThreadLocalMap.Entry e = map.getEntry(this);          if (e != null) {              @SuppressWarnings("unchecked")              T result = (T)e.value;              return result;          }      }      return setInitialValue();  }

其中getMap的源碼:

123
ThreadLocalMap getMap(Thread t) {    return t.threadLocals;}

setInitialValue函數的源碼:

12345678910
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;}

createMap函數的源碼:

123
void createMap(Thread t, T firstValue) {    t.threadLocals = new ThreadLocalMap(this, firstValue);}

簡單解析一下,get方法的流程是這樣的:

首先獲取當前線程根據當前線程獲取一個Map如果獲取的Map不為空,則在Map中以ThreadLocal的引用作為key來在Map中獲取對應的value e,否則轉到5如果e不為null,則返回e.value,否則轉到5Map為空或者e為空,則通過initialValue函數獲取初始值value,然后用ThreadLocal的引用和value作為firstKey和firstValue創建一個新的Map

然后需要注意的是Thread類中包含一個成員變量:

1
ThreadLocal.ThreadLocalMap threadLocals = null;

所以,可以總結一下ThreadLocal的設計思路:每個Thread維護一個ThreadLocalMap映射表,這個映射表的key是ThreadLocal實例本身,value是真正需要存儲的Object。這個方案剛好與我們開始說的簡單的設計方案相反。查閱了一下資料,這樣設計的主要有以下幾點優勢:

這樣設計之后每個Map的Entry數量變小了:之前是Thread的數量,現在是ThreadLocal的數量,能提高性能,據說性能的提升不是一點兩點(沒有親測)當Thread銷毀之后對應的ThreadLocalMap也就隨之銷毀了,能減少內存使用量。

再深入一點

先交代一個事實:ThreadLocalMap是使用ThreadLocal的弱引用作為Key的

12345678910111213141516171819202122
static class ThreadLocalMap {        /**         * The entries in this hash map extend WeakReference, using         * its main ref field as the key (which is always a         * ThreadLocal object).  Note that null keys (i.e. entry.get()         * == null) mean that the key is no longer referenced, so the         * entry can be expunged from table.  Such entries are referred to         * as "stale entries" in the code that follows.         */        static class Entry extends WeakReference<ThreadLocal<?>> {            /** The value associated with this ThreadLocal. */            Object value;            Entry(ThreadLocal<?> k, Object v) {                super(k);                value = v;            }        }        ...        ...}

下圖是本文介紹到的一些對象之間的引用關系圖,實線表示強引用,虛線表示弱引用:

然后網上就傳言,ThreadLocal會引發內存泄露,他們的理由是這樣的:

如上圖,ThreadLocalMap使用ThreadLocal的弱引用作為key,如果一個ThreadLocal沒有外部強引用引用他,那么系統gc的時候,這個ThreadLocal勢必會被回收,這樣一來,ThreadLocalMap中就會出現key為null的Entry,就沒有辦法訪問這些key為null的Entry的value,如果當前線程再遲遲不結束的話,這些key為null的Entry的value就會一直存在一條強引用鏈:Thread Ref -> Thread -> ThreaLocalMap -> Entry -> value永遠無法回收,造成內存泄露。

我們來看看到底會不會出現這種情況。其實,在JDK的ThreadLocalMap的設計中已經考慮到這種情況,也加上了一些防護措施,下面是ThreadLocalMap的getEntry方法的源碼:

12345678
private Entry getEntry(ThreadLocal<?> key) {    int i = key.threadLocalHashCode & (table.length - 1);    Entry e = table[i];    if (e != null && e.get() == key)        return e;    else        return getEntryAfterMiss(key, i, e);}

getEntryAfterMiss函數的源碼:

12345678910111213141516
private Entry getEntryAfterMiss(ThreadLocal<?> key, int i, Entry e) {     Entry[] tab = table;     int len = tab.length;     while (e != null) {         ThreadLocal<?> k = e.get();         if (k == key)             return e;         if (k == null)             expungeStaleEntry(i);         else             i = nextIndex(i, len);         e = tab[i];     }     return null; }

expungeStaleEntry函數的源碼:

1234567891011121314151617181920212223242526272829303132333435
private int expungeStaleEntry(int staleSlot) {           Entry[] tab = table;           int len = tab.length;           // expunge entry at staleSlot           tab[staleSlot].value = null;           tab[staleSlot] = null;           size--;           // Rehash until we encounter null           Entry e;           int i;           for (i = nextIndex(staleSlot, len);                (e = tab[i]) != null;                i = nextIndex(i, len)) {               ThreadLocal<?> k = e.get();               if (k == null) {                   e.value = null;                   tab[i] = null;                   size--;               } else {                   int h = k.threadLocalHashCode & (len - 1);                   if (h != i) {                       tab[i] = null;                       // Unlike Knuth 6.4 Algorithm R, we must scan until                       // null because multiple entries could have been stale.                       while (tab[h] != null)                           h = nextIndex(h, len);                       tab[h] = e;                   }               }           }           return i;       }

整理一下ThreadLocalMap的getEntry函數的流程:

首先從ThreadLocal的直接索引位置(通過ThreadLocal.threadLocalHashCode & (len-1)運算得到)獲取Entry e,如果e不為null并且key相同則返回e;如果e為null或者key不一致則向下一個位置查詢,如果下一個位置的key和當前需要查詢的key相等,則返回對應的Entry,否則,如果key值為null,則擦除該位置的Entry,否則繼續向下一個位置查詢

在這個過程中遇到的key為null的Entry都會被擦除,那么Entry內的value也就沒有強引用鏈,自然會被回收。仔細研究代碼可以發現,set操作也有類似的思想,將key為null的這些Entry都刪除,防止內存泄露。但是光這樣還是不夠的,上面的設計思路依賴一個前提條件:要調用ThreadLocalMap的getEntry函數或者set函數。這當然是不可能任何情況都成立的,所以很多情況下需要使用者手動調用ThreadLocal的remove函數,手動刪除不再需要的ThreadLocal,防止內存泄露。所以JDK建議將ThreadLocal變量定義成private static的,這樣的話ThreadLocal的生命周期就更長,由于一直存在ThreadLocal的強引用,所以ThreadLocal也就不會被回收,也就能保證任何時候都能根據ThreadLocal的弱引用訪問到Entry的value值,然后remove它,防止內存泄露。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
日韩精品免费一线在线观看| 欧美老女人性视频| 欧美日韩亚洲网| 久久av在线看| 91视频国产精品| 国产精品视频资源| 亚洲97在线观看| 欧美视频免费在线观看| 国产精品色午夜在线观看| 九九精品在线播放| 色偷偷偷综合中文字幕;dd| 伊人伊人伊人久久| 91精品国产99| 狠狠做深爱婷婷久久综合一区| 国产精品丝袜久久久久久不卡| 亚洲美女又黄又爽在线观看| 亚洲免费影视第一页| 国产女精品视频网站免费| 九九热这里只有在线精品视| 国产福利成人在线| 国产香蕉精品视频一区二区三区| 亚洲第一色中文字幕| 欧美视频在线观看免费| 孩xxxx性bbbb欧美| 欧美老女人性视频| 国产综合色香蕉精品| 国产这里只有精品| 欧美极品在线播放| 亚洲黄色免费三级| 国产精品入口日韩视频大尺度| 亚洲精品suv精品一区二区| 久久精品国产96久久久香蕉| 91国产视频在线播放| 久久精品在线播放| 国产成人精品视频在线| 国产成人aa精品一区在线播放| 亚洲综合大片69999| 日韩电影免费在线观看中文字幕| 欧美精品成人91久久久久久久| 国产成人精品久久亚洲高清不卡| 久久精品99国产精品酒店日本| 日韩av大片在线| 日韩中文字幕在线精品| 欲色天天网综合久久| 欧美激情在线一区| 久久免费视频网站| 国产成人综合精品在线| 亚洲欧美日韩一区二区在线| 成人亲热视频网站| 国产精品揄拍500视频| 日韩在线观看免费全| 清纯唯美日韩制服另类| 一道本无吗dⅴd在线播放一区| 欧美日韩性生活视频| 欧美在线观看www| 欧美黑人性生活视频| 欧美国产第二页| 欧美超级乱淫片喷水| 在线不卡国产精品| 亚洲国产日韩欧美在线图片| 国产成人在线亚洲欧美| 欧美日韩国产一区在线| 亚洲人成网在线播放| 在线性视频日韩欧美| 国产成人精品日本亚洲| 亚洲欧洲日本专区| 国产一区二区三区在线看| 性夜试看影院91社区| 欧美一区在线直播| 亚洲精品视频中文字幕| 亚洲人成电影网站色xx| 欧美日韩国产综合新一区| 国产精品观看在线亚洲人成网| 欧美激情欧美狂野欧美精品| 91精品视频网站| 日韩在线欧美在线国产在线| 国产一区二区三区在线观看视频| 久久久精品欧美| 欧美日韩激情美女| 最近2019中文字幕大全第二页| 国产成人精品一区二区三区| 欧美国产日韩二区| 欧美精品videossex性护士| 久久国产精品电影| 亚洲欧美日韩一区二区在线| 国产精品ⅴa在线观看h| 精品国产福利在线| 欧美视频在线免费| 日韩专区在线播放| www欧美xxxx| 高潮白浆女日韩av免费看| 中文字幕一区二区精品| 国产精品久久久久久久久男| 欧美午夜视频在线观看| 国产精品老女人精品视频| 成人亲热视频网站| 欧洲日本亚洲国产区| 日韩的一区二区| 色悠悠久久久久| 亚洲欧美中文日韩v在线观看| 欧美一区三区三区高中清蜜桃| 亚洲风情亚aⅴ在线发布| 欧美成人精品在线观看| 亚洲第一综合天堂另类专| 久久在线精品视频| 日本91av在线播放| 久久久久久成人精品| 日韩欧美成人区| 91精品久久久久久综合乱菊| 国产精品va在线| 91精品在线观| 欧美多人乱p欧美4p久久| 亚洲第一区第一页| 综合国产在线观看| 亚洲欧美国产精品va在线观看| 成人女保姆的销魂服务| 亚洲人成网站免费播放| 亚洲嫩模很污视频| 亚洲二区中文字幕| 国产欧美精品日韩| 国产精品无av码在线观看| 亚洲激情 国产| 最新亚洲国产精品| 中文字幕在线看视频国产欧美在线看完整| 欧美激情欧美激情在线五月| 久久久久久久国产精品视频| 欧美日韩国产精品一区二区不卡中文| 精品久久中文字幕| 国产精品久久久久久久久久新婚| 亚洲免费视频在线观看| 美女国内精品自产拍在线播放| 日韩电视剧免费观看网站| 久久精品国产欧美亚洲人人爽| 国产主播在线一区| 91免费看片在线| 尤物精品国产第一福利三区| 欧美黑人一区二区三区| 国产精品成人免费电影| 久久人人爽国产| 日韩极品精品视频免费观看| 亚洲一区二区久久| 成人观看高清在线观看免费| 国产精品久久久久久亚洲影视| 欧美日韩性视频在线| 欧美超级乱淫片喷水| 日产日韩在线亚洲欧美| 国产欧美va欧美va香蕉在线| 91黄色8090| 日韩视频第一页| 国产一区二区欧美日韩| 国产原创欧美精品| 国产成人精品a视频一区www| 91精品国产乱码久久久久久蜜臀| 日韩美女av在线免费观看| 日韩av电影中文字幕| 色综合久久精品亚洲国产| 亚洲精品国产精品久久清纯直播| 一本一道久久a久久精品逆3p| 红桃视频成人在线观看| 91在线观看免费高清| 欧美亚洲视频一区二区| 亚洲欧美日本伦理| 91亚洲精品在线观看| 国产日韩欧美影视|