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

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

上一篇我们分析了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 sStrThreadlocal = new ThreadLocal<>();
private static ThreadLocal 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() {
//先获取当前线程

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!**

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

  • 15
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值