什么是ThreadLocal?
从 Java 官方文档中的描述:ThreadLocal 类用来提供线程内部的局部变量。这种变量在多线程环境下访问(通过get 和 set 方法访问)时能保证各个线程的变量相对独立于其他线程内的变量。ThreadLocal 实例通常来说都是 private static 类型的,用于关联线程和线程上下文。
我们可以得知 ThreadLocal 的作用是:提供线程内的局部变量,不同的线程之间不会相互干扰,这种变量在线程的生命周期内起作用,减少同一个线程内多个函数或组件之间一些公共变量传递的复杂度。
- 线程并发:在多线程并发的场景下
- 传递数据:我们可以通过 ThreadLocal 在同一线程,不同组件之间传递公共变量(有点类似于 Session?)
- 线程隔离:每个线程的变量都是独立的,不会互相影响
基本使用
在介绍 ThreadLocal 使用之前,我们首先认识几个 ThreadLocal 的常见方法
使用案例
我们来看下面这个线程不安全的案例,感受一下 ThreadLocal 线程隔离的特点。
/**
* 需求:线程隔离
* 在多线程并发的场景下,每个线程中的变量都是相互独立的
* 线程A:设置变量1,获取变量2
* 线程B:设置变量2,获取变量2
* @author: 陌溪
*/
public class MyDemo01 {
// 变量
private String content;
public String getContent() {
return content;
}
public void setContent(String content) {
this.content = content;
}
public static void main(String[] args) {
MyDemo01 myDemo01 = new MyDemo01();
for (int i = 0; i < 5; i++) {
new Thread(() -> {
myDemo01.setContent(Thread.currentThread().getName() + "的数据");
System.out.println("-----------------------------------------");
System.out.println(Thread.currentThread().getName() + "\t " + myDemo01.getContent());
}, String.valueOf(i)).start();
}
}
}
运行后的效果
-----------------------------------------
-----------------------------------------
-----------------------------------------
3 4的数据
-----------------------------------------
2 4的数据
-----------------------------------------
1 4的数据
4 4的数据
0 4的数据
从上面我们可以看到,出现了线程不隔离的问题,也就是线程1取出了线程4的内容,那么如何解决呢?
这个时候就可以用到 ThreadLocal 了,我们通过 set 将变量绑定到当前线程中,然后get 获取当前线程绑定的变量
/**
* 需求:线程隔离
* 在多线程并发的场景下,每个线程中的变量都是相互独立的
* 线程A:设置变量1,获取变量2
* 线程B:设置变量2,获取变量2
* @author: 陌溪
*/
public class MyDemo01 {
// 变量
private String content;
public String getContent() {
return content;
}
public void setContent(String content) {
this.content = content;
}
public static void main(String[] args) {
MyDemo01 myDemo01 = new MyDemo01();
ThreadLocal<String> threadLocal = new ThreadLocal<>();
for (int i = 0; i < 5; i++) {
new Thread(() -> {
threadLocal.set(Thread.currentThread().getName() + "的数据");
System.out.println("-----------------------------------------");
System.out.println(Thread.currentThread().getName() + "\t " + threadLocal.get());
}, String.valueOf(i)).start();
}
}
}
通过引入 ThreadLocal 后,查看运行结果如下:
-----------------------------------------
-----------------------------------------
4 4的数据
-----------------------------------------
3 3的数据
-----------------