Android基础进阶 - 消息机制 之ThreadLocal

目录

  1. ThreadLocal是什么
  2. 如何使用ThreadLocal
  3. ThreadLocal源码分析
  4. 消息机制中Looper中的ThreadLocal使用
  5. 资料
  6. 收获

上一篇我们分析了Anrdoid消息机制的实现,其中关于ThreadLocal以及Native层的还没有搞清楚,这篇我们来一起学习分析下ThreadLocal的作用。

一、ThreadLocal是什么

ThreadLocal 线程局部变量 是一个泛型类,可以接受任何类型的对象,一般ThreadLocal的类型的变量时static类型的。

我们知道不同线程有自己的栈,但是内存资源在同一个进程是共享的,即不同线程可以访问同一个变量,这样就会有多线程同步问题。即一个线程修改了变量,另外一个线程再读,如果不加锁或者volatile,可能导致不同线程的获取的结果不一致。

想象下面一种场景:满足下面两个条件

  1. 一个对象中的一个变量,会在不同的方法中会使用,这个对象会在不同线程中调用。
  2. 这个变量不需要多线程同步,而是需要每个线程都一份独立的值,即是线程隔离的。

这是我们该如何设计呐?
可能我们首先想到的是通过Map的方式,Key来存储Thread(eg:ThreadId),Value来存储每个Thread中该变量的值。在对map的读写操作上加上同步锁,即可实现上面场景的需求。
但是这种方案由于加了锁,会带来一定的性能损耗,是否还有更好的方式来实现线程隔离呐?

今天分析的ThreadLocal就是为此而设计的,它适用于每个线程需要自己独立的实例且该实例需要在多个方法中被使用,也即变量在线程间隔离而在方法或类间共享的场景。

使用ThreadLocal修饰的变量,在每个线程内都有自己副本,且该副本只能在自己的线程使用,实现了线程隔离。

二、如何使用ThreadLocal

这一小节,我们通过一个简单测试代码来说明ThreadLocal使用和验证它的线程隔离的特性。

在一个类中定义ThreadLocal类型的变量,分别在不同的线程赋不同的值,然后输出看下不同线程之间是否有影响。

public class ExampleUnitTest {

    //定义两个不同类型的ThreadLocal
    private static ThreadLocal<String> sStrThreadlocal = new ThreadLocal<>();
    private static ThreadLocal<Integer> sIntegerThreadLocal = new ThreadLocal<>();
    @Test
    public void addition_isCorrect() {
        assertEquals(4, 2 + 2);
    }

    @Test
    public void testThreadLocal(){

        //在主线程给Threadlocal赋值,请取出输出

        sStrThreadlocal.set("aaa");
        String value = sStrThreadlocal.get();

        sIntegerThreadLocal.set(1);
        int intValue = sIntegerThreadLocal.get();
        System.out.println("111 curThreadId="+Thread.currentThread()+" strthreadLocalValue="+value
        +" intThreadLocalValue="+intValue);

        //创建两个线程,分别给ThreadLocal赋不同的值
        new Thread(new Runnable() {
            @Override
            public void run() {
                sStrThreadlocal.set("bbb");
                String value = sStrThreadlocal.get();

                sIntegerThreadLocal.set(2);
                int intValue = sIntegerThreadLocal.get();
                System.out.println("222 curThreadId="+Thread.currentThread()+" strthreadLocalValue="+value
                        +" intThreadLocalValue="+intValue);

            }
        }).start();

        new Thread(new Runnable() {
            @Override
            public void run() {
                String value = sStrThreadlocal.get();

                Integer intValue = sIntegerThreadLocal.get();

                System.out.println("333 curThreadId="+Thread.currentThread()+" strthreadLocalValue="+value
                        +" intThreadLocalValue="+intValue);

            }
        }).start();

        try {
            Thread.sleep(100);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        //最后在输出下主线程的ThreadLocal值
        value = sStrThreadlocal.get();
        intValue = sIntegerThreadLocal.get();
        System.out.println("444 curThreadId="+Thread.currentThread()+" strthreadLocalValue="+value
                +" intThreadLocalValue="+intValue);

    }
}

运行结果如下:

111 curThreadId=Thread[main,5,main] strthreadLocalValue=aaa intThreadLocalValue=1
222 curThreadId=Thread[Thread-0,5,main] strthreadLocalValue=bbb intThreadLocalValue=2
333 curThreadId=Thread[Thread-1,5,main] strthreadLocalValue=null intThreadLocalValue=null
444 curThreadId=Thread[main,5,main] strthreadLocalValue=aaa intThreadLocalValue=1

不同线程给ThreadLocal修饰的变量赋不同的值,在每个线程得到的值不同的,的确实现了线程的隔离。

那么它是如何做到的呐?是否是通过HashMap来存储不同线程的value值呐?我们通过分析ThreadLocal源码来找下答案。

三、ThreadLocal源码分析

ThreadLocal 是一个泛型类,可以接受任何类型的对象

正如上面的示例代码所示,一个线程内可以存在多个 ThreadLocal 对象,而ThreadLocal 内部维护了一个 Map ,满足这种需求。
但是这个 Map 不是直接使用的 HashMap ,而是 ThreadLocal 实现的一个叫做 ThreadLocalMap 的静态内部类

通过上面示例我们可以看到 通过set方式给ThreadLocal设置数据,get方法获取数据,我们以此为入口来进行分析

ThreadLocal#set

    public void set(T value) {
        //获取调用方所在的线程
        Thread t = Thread.currentThread();
        //获取该线程的ThreadLocal的副本,这个getMap方法是关键
        ThreadLocalMap map = getMap(t);
        //如果该线程存在该ThreadLocal的副本,则存入到map中,key,否则创建
        if (map != null)
            map.set(this, value);
        else
            createMap(t, value);
    }

getMap

获取该线程的ThreadLocal的副本
    ThreadLocalMap getMap(Thread t) {
        return t.threadLocals;
    }

来看下Thread类,发现threadLocals变量的类型是ThreadLocal.ThreadLocalMap,即ThreadLocal的一个静态内部类

每个Thread对象内部都维护了一个ThreadLocalMap, 其可以存放若干个ThreadLocal

public class Thread implements Runnable {
    ......
    //当前线程的ThreadLocalMap,主要存储该线程自身的ThreadLocal,本文主要讨论这个变量
    ThreadLocal.ThreadLocalMap threadLocals = null;
    //自父线程继承而来的ThreadLocalMap,主要用于父子线程间ThreadLocal变量的传递
    ThreadLocal.ThreadLocalMap inheritableThreadLocals = null;
    .....
}

再来看下ThreadLocal.ThreadLocalMap

   static class ThreadLocalMap {
        .......
        private ThreadLocal.ThreadLocalMap.Entry[] table;

ThreadLocal.ThreadLocalMap.Entry

Entry的key是ThreadLocal的弱引用,value是对应的线程中线程局部变量set的值。
我们知道弱引用在GC的时候会销毁该引用所包裹(引用)的对象,这个threadLocal作为key可能被销毁(如果没有强引用存在),如果key为空,则该entry会从table中删除

static class Entry extends WeakReference<ThreadLocal<?>> {
            Object value;

            Entry(ThreadLocal<?> var1, Object var2) {
                super(var1);
                this.value = var2;
            }
        }


图片来自深入解析ThreadLocal 详解、实现原理、使用场景方法以及内存泄漏防范 多线程中篇(十七)

从本质来讲,就是每个线程都维护了一个map,而这个map的key就是threadLocal,而值就是我们set的那个值

分析完了set链路,我们再来看下get链路

当我们在调用get()方法的时候,先获取当前线程,然后获取到当前线程的ThreadLocalMap对象,如果非空,那么取出ThreadLocal的value,否则进行初始化,初始化就是将initialValue的值set到ThreadLocal中。

public T get() {
        //先获取当前线程
        Thread t = Thread.currentThread();
        //获取到当前线程的ThreadLocalMap对象
        ThreadLocalMap map = getMap(t);
        if (map != null) {
            //如果非空,那么取出ThreadLocal的value
            ThreadLocalMap.Entry e = map.getEntry(this);
            if (e != null) {
                T result = (T)e.value;
                return result;
            }
        }
        //否则进行初始化,初始化就是将initialValue的值set到ThreadLocal中
        return setInitialValue();
    }

四、消息机制中Looper中的ThreadLocal使用

在Android的Framework中很多地方都使用了ThreadLocal,比如Looper、Choreographer、ActivityThread、ContentProvide、ViewRootImpl、SQLiteDatabase等等,在调用链追踪方面也是可以使用。
我们来分析下每个线程的Looper保证独立,并且一个线程有且只有一个Looper的

public final class Looper {
    ......
    // sThreadLocal.get() 将会返回 null,直到调用了 prepare().
    // sThreadLocal是一个ThreadLocal的一个实例,其类型为Looper
    static final ThreadLocal<Looper> sThreadLocal = new ThreadLocal<Looper>();


    final MessageQueue mQueue;
    ......
}

通过ThreadLocal的线程隔离 保证每个线程的Looper是不同的,
通过sThreadLocal.get() != null的异常断言,保证了一个线程只能有一个Looper
//初始化,将当前线程初始化为循环器

private static void prepare(boolean quitAllowed) {
        //通过ThreadLocal的线程隔离 保证每个线程的Looper是不同的,
        //通过sThreadLocal.get() != null的异常断言,保证了一个线程只能有一个Looper
        if (sThreadLocal.get() != null) {
            throw new RuntimeException("Only one Looper may be created per thread");
        }
        sThreadLocal.set(new Looper(quitAllowed));
    }

    /**
      * 返回调用该方法所在线程对应的Looper
      * 如果调用者的线程还没有和Looper关联(通过preprea),则返回空
     */
    public static @Nullable Looper myLooper() {
        return sThreadLocal.get();
    }

五、资料

  1. Android源码

  2. ThreadLocal详解

  3. InheritableThreadLocal详解

  4. ThreadLocal原理分析与使用场景

  5. 自信,这是最好的ThreadLocal分析

  6. IO多路复用之epoll总结

  7. Android 中 MessageQueue 的 nativePollOnce

  8. Class 'kotlin.Unit' was compiled with an incompatible version of Kotlin

  9. 深入解析ThreadLocal 详解、实现原理、使用场景方法以及内存泄漏防范 多线程中篇(十七)

六、收获

通过对本篇的学习实践

  1. 了解了ThreadLocal的意义以及原理
  2. ThreadLocal的使用场景,它适用于每个线程需要自己独立的实例且该实例需要在多个方法中被使用,也即变量在线程间隔离而在方法或类间共享的场景。
  3. Android消息机制中通过ThreadLocal保证Looper的线程隔离,get是断言判断保证一个Thread只能有一个Looper。

感谢你的阅读
下一篇我们分析消息机制的Native层,分析了解是如何阻塞和唤醒的,欢迎关注公众号“音视频开发之旅”,一起学习成长。
欢迎交流

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

推荐阅读更多精彩内容