ThreadLocal源码解析

ThreadLocal

1. ThreadLocal 是什么?

/**
 * Implements a thread-local storage, that is, a variable for which each thread
 * has its own value. All threads share the same {@code ThreadLocal} object,
 * but each sees a different value when accessing it, and changes made by one
 * thread do not affect the other threads. The implementation supports
 * {@code null} values.
 * @see java.lang.Thread
 * @author Bob Lee
 */

源码中解释到 ThreadLocal 就用于存储数据,每一个线程都会一个数据的拷贝,每一个线程获取到值都是不一样的,并且当前线程修改之后,不会应到其他线程。

2. ThreadLocal 的作用?

这里举例解释 ThreadLoal 在andorid 源码的应用。使用Handler就必须获取当前线程的 Looper 对象,而每一个线程的 Looper 是不一致的。因为每一个线程都会有一个 Looper 对象,因此使用 ThradLocal 去保存和获取当前线程的 Looper 就可以达到这个的效果。

2.1.下面就是 Looper 内部的关于在 ThreadLocal 中存储 Looper 和 获取 Looper 的源码。

//创建一个 Looper 对象保存到 ThreadLocal
//Looper.prepare();

private static void prepare(boolean quitAllowed) {
    if (sThreadLocal.get() != null) {
        throw new RuntimeException("Only one Looper may be created per thread");
    }
    //将创建的 Looper 对象保存到 sThreadLocal 中。
    sThreadLocal.set(new Looper(quitAllowed));
}


//从 ThreadLocal 取出 Looper 对象
public static @Nullable Looper myLooper() {
    return sThreadLocal.get();
}

3. 下面举例说明 ThreadLocal 存储的数据在各个线程中是不会相互干扰的。

// 创建一个ThreadLocal
final ThreadLocal<String> threadLocal = new ThreadLocal<>();
//在主线程中设置threadlocal的值为"hello"
threadLocal.set("hello");
String msg = threadLocal.get();
Log.e("zeal","主线程:"+ msg);
//在子线程中操作
new Thread(){
    @Override
    public void run() {
        super.run();
        threadLocal.set("android");
        String msg = threadLocal.get();
        Log.e("zeal", "线程"+Thread.currentThread().getName()+":" + msg);
    }
}.start();
//在子线程中操作
new Thread(){
    @Override
    public void run() {
        super.run();
        String msg = threadLocal.get();
        Log.e("zeal", "线程"+Thread.currentThread().getName()+":" + msg);
    }
}.start();

运行结果:
主线程:hello
线程Thread-172:android
线程Thread-173:null

运行结果说明不同线程取出之前保存的值是不相同的,也就是即使在某一线程修改了这个ThreadLocal保存的值,也是不会相互干扰到其他线程的。

4. ThreadLocal 的内部实现原理

4.1 通过 set 为ThreadLocal 设置值

将当前线程封装成一个 Values 对象,然后以 ThreadLocal 为 key, 需要添加的value 为值添加到 values 中的。注意在 put 方法中 ThreadLoca 的值是存储在 ThreadLoal 内部一个 table 数组中的,并且存储位置为 ThreadLocal.Reference 角标的下一个位置。

public void set(T value) {
    Thread currentThread = Thread.currentThread();
    Values values = values(currentThread);
    if (values == null) {
        values = initializeValues(currentThread);
    }
    values.put(this, value);
}

void put(ThreadLocal<?> key, Object value) {
    ...
    table[index] = key.reference;
    table[index + 1] = value;
    size++;
    ...
}
Values values(Thread current) {
    return current.localValues;
}
Values initializeValues(Thread current) {
    return current.localValues = new Values();
}

4.2 通过 get 的方式获取保存的 ThreadLocal 值

获取当前的线程对应的Values,其实这个 Values 就是 Thread 中的一个属性值 ThreadLocal.Values localValues,如果为null,那么通过initializeValues(currentThread) 给当前线程的 localValues 赋值。否则获取存储 ThreadLocal 值的 table 数组,判断table中是否存在 ThreadLocal 设置的值。如果没有则返回null;

public T get() {
    // Optimized for the fast path.
    Thread currentThread = Thread.currentThread();
    Values values = values(currentThread);
    if (values != null) {
        Object[] table = values.table;
        int index = hash & values.mask;
        if (this.reference == table[index]) {
            return (T) table[index + 1];
        }
    } else {
        values = initializeValues(currentThread);
    }
    return (T) values.getAfterMiss(this);
}

5. 得出的结论

ThreadLocal 中 set 和 get 操作的都是对应线程的 table数组,因此在不同的线程中访问同一个 ThreadLocal 对象的 set 和 get 进行存取数据是不会相互干扰的。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,142评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,298评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,068评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,081评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,099评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,071评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,990评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,832评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,274评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,488评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,649评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,378评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,979评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,625评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,643评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,545评论 2 352

推荐阅读更多精彩内容