ThreadLocal就是一个类,他有get、set方法,可以起到一个保存、获取某个值的作用。但是这个类的get、set方法有点特殊,各个线程调用时是互不干扰的,就好像线程在操作ThreadLocal对象时是在操作线程自己的私有属性一样。具体原因在于他的方法实现:
publicTget(){Threadt=Thread.currentThread();//先确定调用我的线程ThreadLocalMapmap=getMap(t);//根据调用我的线程,找到这个线程的ThreadLocalMap对象if(map!=null){ThreadLocalMap.Entrye=map.getEntry(this);//以ThreadLocal对象为key,找到对应元素if(e!=null){@SuppressWarnings("unchecked")Tresult=(T)e.value;//讲元素的value返回returnresult;}}returnsetInitialValue();//如果调用我的线程没有ThreadLocalMap对象,则返回初始值}publicvoidset(Tvalue){Threadt=Thread.currentThread();//先确定调用我的是哪个线程ThreadLocalMapmap=getMap(t);//获取调用我的线程的ThreadLocalMapif(map!=null)map.set(this,value);//如果那个线程有map,就将此ThreadLocal对象为key的value设置好elsecreateMap(t,value);//如果那个线程还没有map,先创建一个再设置}ThreadLocalMap是ThreadLocal的内部类,为了不造成混乱,可以把他看作一个普通的类。ThreadLocalMap其实类似与HashMap,也是通过key获取某个值(key就是ThreadLocal对象),也是数组存储键值对,拉链法解决冲突等。一个Thread类持有一个ThreadLocalMap实例。
通过上面的源码也可以看出:线程互不干扰的操作ThreadLocal的原因就是,它的set、get方法是要先获取当前线程,然后修改、操作这个线程对象的成员属性。也就是说,调用ThreadLocal对象的set、get方法实际上是在操作当前线程的成员属性,只不过这些属性是通过ThreadLocal对象为key找到的而已。为了值观明了,看下图:
简单概括过程:有ThreadLocal对象tl,线程t调用tl.get(),则去线程t的ThreadLocalMap属性对象里找到一个entry,若entry.key==tl返回true,则此entry是目标entry,此entry.value就是我们的目标。
ThreadLocal对象只是一个获取当前线程某个私有属性的渠道而已,提供了set、get的入口,同时作为key去获取和设置目标值。真正的有效目标是属于线程对象私自持有的,自然通过ThreadLocal对象获取的值也就不会受其他线程影响啦。
二.用法示例
publicclassThreadLocalTest{privatestaticThreadLocal
理解了ThreadLocal的原理,使用起来很简单,注意ThreadLocal对象的定义位置,检查作用域,保证可以被要使用它的线程访问到。
三.关于Entry的弱类型引用
如果阅读ThreadLocalMap的Entry源码会发现,Entry的key是弱引用:
staticclassEntryextendsWeakReference
publicvoidfunc1(){ThreadLocaltl=newThreadLocal
当func1方法执行完毕后,栈帧销毁,强引用tl也就没有了,但此时线程的ThreadLocalMap里某个entry的k引用还指向这个对象。若这个k引用是强引用,就会导致k指向的ThreadLocal对象及v指向的对象不能被gc回收,造成内存泄漏,但是弱引用就不会有这个问题(弱引用及强引用等这里不说了)。使用弱引用,就可以使ThreadLocal对象在方法执行完毕后顺利被回收,而且在entry的k引用为null后,再调用get,set或remove方法时,就会尝试删除key为null的entry,可以释放value对象所占用的内存。
概括说就是:在方法中新建一个ThreadLocal对象,就有一个强引用指向它,在调用set()后,线程的ThreadLocalMap对象里的Entry对象又有一个引用k指向它。如果后面这个引用k是强引用就会使方法执行完,栈帧中的强引用销毁了,对象还不能回收,造成严重的内存泄露。
注意:虽然弱引用,保证了k指向的ThreadLocal对象能被及时回收,但是v指向的value对象是需要ThreadLocalMap调用get、set时发现k为null时才会去回收整个entry、value,因此弱引用不能保证内存完全不泄露。我们要在不使用某个ThreadLocal对象后,手动调用remoev方法来删除它,尤其是在线程池中,不仅仅是内存泄露的问题,因为线程池中的线程是重复使用的,意味着这个线程的ThreadLocalMap对象也是重复使用的,如果我们不手动调用remove方法,那么该线程后面执行时可能获取到上次执行遗留下来的value值,造成bug。
四.用途举例
Spring的RequestContextHolder就是这么操作的,这样在使用切面时,也可以获取到请求信息了(切面编程时自身是只可以获取到方法名+方法参数信息的):