理解和使用ThreadLocal类

一、从数据结构入手

  下图为ThreadLocal的内部结构图

  

  从上面的机构图,可以窥见ThreadLocal的核心机制:

  • 每个Thread线程内部都有一个Map;
  • Map里面存储线程本地对象(key)和线程的变量副本(value)
  • 但是,Thread内部的Map是由ThreadLocal维护的,由ThreadLocal负责向map获取和设置线程的变量值

  所以对于不同的线程,每次获取副本值时,别的线程并不能获取到当前线程的副本值,形成了副本的隔离,互不干扰。

 

二、深入解析ThreadLocal

  ThreadLocal类提供如下几个核心方法:

public T get()
public void set(T value)
public void remove()
protected T initialValue() { }
  • get()方法用于获取当前线程的变量副本
  • set()方法用户保存当前线程的变量副本
  • initialValue()是一个protected方法,一般是用来在使用时进行重写的,它是一个延迟加载方法,用于为当前线程初始变量副本值
  • remove()方法移除当前线程的副本变量

  get()方法

 1 /**
 2  * Returns the value in the current thread's copy of this
 3  * thread-local variable.  If the variable has no value for the
 4  * current thread, it is first initialized to the value returned
 5  * by an invocation of the {@link #initialValue} method.
 6  *
 7  * @return the current thread's value of this thread-local
 8  */
 9 public T get() {
10     Thread t = Thread.currentThread();
11     ThreadLocalMap map = getMap(t);
12     if (map != null) {
13         ThreadLocalMap.Entry e = map.getEntry(this);
14         if (e != null)
15             return (T)e.value;
16     }
17     return setInitialValue();
18 }

  第一句取得当前线程,然后通过getMap(t)方法获取到一个map,map的类型为ThreadLocalMap。然后接着下面获取到<key,value>键值对Entry节点,注意这里获取Entry传进去的是this,而不是当前线程t

  如果获取成功,则返回value值。

  如果map为空,则调用setInitialValue方法返回value。

  我们对上面的每一句仔细分析:

  首先看一下getMap方法中做了什么:

 1    /**
 2      * Get the map associated with a ThreadLocal. Overridden in
 3      * InheritableThreadLocal.
 4      *
 5      * @param  t the current thread
 6      * @return the map
 7      */
 8     ThreadLocalMap getMap(Thread t) {
 9         return t.threadLocals;
10     }

  可能大家没有想到的是,在getMap中,是调用当前线程t,返回当前线程t中的一个成员变量threadLocals

  那么我们继续取Thread类中去看一下成员变量threadLocals是什么:

1 public class Thread implements Runnable {
2     
3     /* ThreadLocal values pertaining to this thread. This map is maintained
4      * by the ThreadLocal class. */
5     ThreadLocal.ThreadLocalMap threadLocals = null;
6 }

  实际上就是一个ThreadLocalMap,这个类型是ThreadLocal类的一个内部类,我们继续去看ThreadLocalMap的实现:

 1    static class ThreadLocalMap {
 2 
 3         /**
 4          * The entries in this hash map extend WeakReference, using
 5          * its main ref field as the key (which is always a
 6          * ThreadLocal object).  Note that null keys (i.e. entry.get()
 7          * == null) mean that the key is no longer referenced, so the
 8          * entry can be expunged from table.  Such entries are referred to
 9          * as "stale entries" in the code that follows.
10          */
11         static class Entry extends WeakReference<ThreadLocal> {
12             /** The value associated with this ThreadLocal. */
13             Object value;
14 
15             Entry(ThreadLocal k, Object v) {
16                 super(k);
17                 value = v;
18             }
19         }
20 }

  可以看到ThreadLocalMap的内部类Entry继承了WeakReference,并且使用ThreadLocal作为键值。

  然后再继续看setInitialValue()方法的具体实现:

 1     /**
 2      * Variant of set() to establish initialValue. Used instead
 3      * of set() in case user has overridden the set() method.
 4      *
 5      * @return the initial value
 6      */
 7     private T setInitialValue() {
 8         T value = initialValue();
 9         Thread t = Thread.currentThread();
10         ThreadLocalMap map = getMap(t);
11         if (map != null)
12             map.set(this, value);
13         else
14             createMap(t, value);
15         return value;
16     }

  很容易了解,就是如果map不为空,就设置键值对,为空,再创建Map,看一下createMap的实现:

 1     /**
 2      * Create the map associated with a ThreadLocal. Overridden in
 3      * InheritableThreadLocal.
 4      *
 5      * @param t the current thread
 6      * @param firstValue value for the initial entry of the map
 7      * @param map the map to store.
 8      */
 9     void createMap(Thread t, T firstValue) {
10         t.threadLocals = new ThreadLocalMap(this, firstValue);
11     }

  至此,我们可以明白ThreadLocal是如何为每个线程创建变量的副本的:

  首先,在每个线程Thread内部有一个ThreadLocal.ThreadLocalMap类型的成员变量threadLocals,这个threadLocals就是用来存储实际的变量副本的,key为当前ThreadLocal变量,value为变量副本(即T类型的变量)。

  初始时,在Thread里面,threadLocals为空,当通过ThreadLocal变量调用get()方法或者set()方法,就会对Thread类中的threadLocals进行初始化,并且以当前ThreadLocal变量为键值,以ThreadLocal要保存的副本变量为value,存到threadLocals。

  然后在当前线程里面,如果要使用副本变量,就可以通过get方法在threadLocals里面查找

  set()方法

 1 /**
 2  * Sets the current thread's copy of this thread-local variable
 3  * to the specified value.  Most subclasses will have no need to
 4  * override this method, relying solely on the {@link #initialValue}
 5  * method to set the values of thread-locals.
 6  *
 7  * @param value the value to be stored in the current thread's copy of
 8  *        this thread-local.
 9  */
10 public void set(T value) {
11     Thread t = Thread.currentThread();
12     ThreadLocalMap map = getMap(t);
13     if (map != null)
14         map.set(this, value);
15     else
16         createMap(t, value);
17 }

  获取当前线程的成员变量map,若map非空,则重新将ThreadLocal和新的value副本放入到map中。

  若map空,则对线程的成员变量ThreadLocalMap进行初始化创建,并将ThreadLocal和value副本放入map中。

  remove()方法

 1 /**
 2  * Removes the current thread's value for this thread-local
 3  * variable.  If this thread-local variable is subsequently
 4  * {@linkplain #get read} by the current thread, its value will be
 5  * reinitialized by invoking its {@link #initialValue} method,
 6  * unless its value is {@linkplain #set set} by the current thread
 7  * in the interim.  This may result in multiple invocations of the
 8  * <tt>initialValue</tt> method in the current thread.
 9  *
10  * @since 1.5
11  */
12 public void remove() {
13  ThreadLocalMap m = getMap(Thread.currentThread());
14  if (m != null)
15      m.remove(this);
16 }

  将当前ThreadLocal对应的value副本从map中移除。

  

 

三、ThreadLocalMap

  ThreadLocalMap是ThreadLocal的内部类,没有实现Map接口,用独立的方式实现了Map的功能,其内部的Entry也独立实现。

  在ThreadLocalMap中,也是用Entry来保存K-V结构数据的。但是Entry中key只能是Threadlocal对象,这点被Entry的构造方法已经限定死了

1 static class Entry extends WeakReference<ThreadLocal> {
2     /** The value associated with this ThreadLocal. */
3     Object value;
4 
5     Entry(ThreadLocal k, Object v) {
6         super(k);
7         value = v;
8     }
9 }

  Entry继承自WeakReference(弱引用,生命周期只能存活到下次GC前),但只有Key是弱引用类型的,Value并非弱引用

  ThreadLocalMap的成员变量:

 1 static class ThreadLocalMap {
 2     /**
 3      * The initial capacity -- MUST be a power of two.
 4      */
 5     private static final int INITIAL_CAPACITY = 16;
 6 
 7     /**
 8      * The table, resized as necessary.
 9      * table.length MUST always be a power of two.
10      */
11     private Entry[] table;
12 
13     /**
14      * The number of entries in the table.
15      */
16     private int size = 0;
17 
18     /**
19      * The next size value at which to resize.
20      */
21     private int threshold; // Default to 0
22 }

 

四、Hash冲突怎么解决

  和HashMap的最大不同在于,ThreadLocalMap结构非常简单,没有next引用,也就是说ThreadLocalMap中解决Hash冲突的方式并非链表的方式,而是采用线性探测的方式,所谓线性探测,就是根据初始Key的hashcode值确定元素在table数组中的位置,如果发现这个位置上已经有其它Key值的元素被占用,则利用固定的算法寻找一定步长的下个位置,依次判断,直至找到能够存放的位置

  ThreadLocalMap解决Hash冲突的方式就是简单的步长加1或减1,寻找下一个相邻的位置。

 1 /**
 2  * Increment i modulo len.
 3  */
 4 private static int nextIndex(int i, int len) {
 5     return ((i + 1 < len) ? i + 1 : 0);
 6 }
 7 
 8 /**
 9  * Decrement i modulo len.
10  */
11 private static int prevIndex(int i, int len) {
12     return ((i - 1 >= 0) ? i - 1 : len - 1);
13 }

  显然ThreadLocalMap采用线性探测的方式解决Hash冲突的效率很低,如果有大量不同的ThreadLocal对象放入map中时发生冲突,或者发生二次冲突,则效率很低。

  所以这里引出的良好建议是:每个线程只存放一个变量,这样的话所有的线程存放到map中的Key都是相同的ThreadLocal,如果一个线程要保存多个变量,就需要创建多个ThreadLocal,多个ThreadLocal放入Map中时会极大地增加Hash冲突的可能

 

五、ThreadLocalMap的问题

  由于ThreadLocalMap的Key是弱引用,而Value是强引用。这就导致一个问题,ThreadLocal在没有外部对象强引用时,发生GC时弱引用Key会被回收,而Value不会回收,如果创建ThreadLocal的线程一直持续运行,那么这个Entry对象中的value就有可能一直得不到回收,发生内存泄漏。

  如何避免泄漏

  既然Key是弱引用,那么我们要做的事,就是在调用ThreadLocal的get()、set()方法时,完成后再调用remove()方法,将Entry节点和Map的引用关系移除,这样整个Entry对象在GC Roots分析后就变成不可达了,下次GC的时候就可以被回收。

  如果使用ThreadLocal的get方法之后,没有显式地调用remove()方法,就有可能发生内存泄漏,所以养成良好的变成习惯十分重要,使用完ThreadLocal之后,记得调用remove()方法。如:

ThreadLocal<Session> threadLocal = new ThreadLocal<Session>();
try {
    threadLocal.get();
    // 其它业务逻辑
} finally {
    threadLocal.remove();
}

 

六、ThreadLocal的应用场景

  最常见的ThreadLocal的使用场景为用来解决数据库连接、Session管理等。

  如:

 1 private static final ThreadLocal threadSession = new ThreadLocal();
 2  
 3 public static Session getSession() throws InfrastructureException {
 4     Session s = (Session) threadSession.get();
 5     try {
 6         if (s == null) {
 7             s = getSessionFactory().openSession();
 8             threadSession.set(s);
 9         }
10     } catch (HibernateException ex) {
11         throw new InfrastructureException(ex);
12     }
13     return s;
14 }

  可以看到,在getSession()方法中,首先判断当前线程中有没有放进去session,如果还没有,那么通过sessionFactory().openSession()来创建一个session,再将session set到线程中,实际上放到当前线程ThreadLocalMap这个map中,这时,对于这个session的唯一引用就是当前线程中的那个ThreadLocalMap,而threadSession作为这个值的Key,要取得这个session可以通过threadSession.get()来得到,里面执行的操作实际上是先取得当前线程的ThreadLocalMap,然后将threadSession作为key将对应的值取出。这个session相当于线程的私有变量。显然,其它线程中是取不到这个session的,他们也只能取到自己的ThreadLocalMap中的东西。

  总之,ThreadLocal不是用来解决对象共享访问问题的,而主要是提供了保持对象的方法和避免参数传递的方便的对象访问方式

  ThreadLocal的应用场合,我觉得最适合的是对同一个线程调用的多个方法中共享了某一个变量(如上面的session),这个变量需要传递到多个方法中(如上的session在service层和dao层的很多方法中都要用到),这样传来传去太麻烦了,这时可以采用ThreadLocal

  这里还有一个在Java Web中使用ThreadLocal的例子可供学习:《优雅的使用ThreadLocal

 

七、总结

  ThreadLocal并不是为了解决线程安全问题,而是提供了一种将实例绑定到当前线程的机制,类似于隔离的效果,实际上自己的方法中new出来变量也能达到类似的效果。ThreadLocal跟线程安全基本不搭边,绑定上去的实例也不是多线程共用的,而是每个线程new一份,这个实例肯定不是共用的,如果共用了,那就会引发线程安全问题。ThreadLocal最大的用处是用来把实例变量共享成全局变量,在线程的任何方法中都可以访问到该实例变量而已。网上很多人说ThreadLocal是解决了线程安全问题,其实是望文生义,两者不是同类问题。

  

 

补充:ThreadLocal类型变量为何声明为静态static?

  Java中每个线程都有与之关联的Thread对象,Thread对象中有一个ThreadLocal.ThreadLocalMap类型的成员变量,该变量是一个Hash表,所以每个线程都单独维护这样一个Hash表,当ThreadLocal类型对象调用 set()方法时,这个set()方法会使用当前线程维护的Hash表,把自己(ThreadLocal)作为Key,相应的值作为value插入到Hash表中。由于每个线程维护的Hash表是独立的,因此在不同的Hash表中,key值即使相同也是没问题的

  如果把ThreadLocal对象声明为非静态的,则当包含ThreadLocal类声明的类去产生一个实例时,都会产生一个ThreadLocal的新对象,这是毫无意义的,只是增加了内存消耗

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值