关于多线程的学习,如果不了解ThreadLocal,那么多少有点说不过去,本文将详细讲述ThreadLocal的各个方面,呕心值作,相信我,看完你一定会有所收获,认真看完的你一定很棒~
ThreadLocal是什么?有什么用?
从ThreadLocal的名字上可以看到,这是一个线程的局部变量。也就是说,只有当前线程可以访问。既然是只有当前线程可以访问的数据,自然是线程安全的。
当使用ThreadLocal维护变量时,ThreadLocal为每个使用该变量的线程提供独立的变量副本,所以每一个线程都可以独立地改变自己的副本,而不会影响其它线程所对应的副本
以JDBC中的一个简单例子来说明如何使用ThreadLocal
当线程A获取到Connection,开启一个事务,正在在执行事务,但是未结束。此时,线程B也获取到Connection,它发送了一些SQL操作,这些SQL操作将会被数据库归入线程A的事务当中被执行,所以JDBC Connection 类是非线程安全的,两个线程不能安全地共享一个 Connection
先别急,Thread,ThreadLocal,ThreadLocalMap,Entry都是什么关系?
以如下代码为例,是一个ThreadLocal的简单使用
public class TestThreadLocal {
public static void main(String[] args) {
ThreadLocal<Person> tl = new ThreadLocal<>();
ThreadLocal<Person> tl1 = new ThreadLocal<>();
tl.set(new Person("jack"));
tl1.set(new Person("rose"));
Person person = tl.get();
Person person1 = tl1.get();
System.out.println(person1);
}
}
首先ThreadLocalMap是以数组形式存储一个个Entry键值对的,它是Thread的一个静态内部类,而Entry是ThreadLocalMap的静态内部类,Entry的key就是new出来的ThreadLocal,value就是set入的值,所以一个Thread可以有多个ThreadLocal-value键值对,而每个Thread只能维护唯一的ThreadLocals
这里我们基本上可以找到ThreadLocal数据隔离的真相了,每个线程Thread都维护了自己的threadLocals变量,所以在每个线程创建ThreadLocal的时候,实际上数据是存在自己线程Thread的threadLocals变量里面的,别人没办法拿到,从而实现了隔离。
public class ThreadLocal<T> {
。。。。
static class ThreadLocalMap {
static class Entry extends WeakReference<ThreadLocal<?>> {
/** The value associated with this ThreadLocal. */
Object value;
//构造方法保证key只能是ThreadLocal值