深入ThreadLocal的内部机制

来源:互联网 发布:淘宝权在哪里直播间 编辑:程序博客网 时间:2024/05/14 06:29

早在JDK 1.2的版本中就提供java.lang.ThreadLocal,ThreadLocal为解决多线程程序的并发问题提供了一种新的思路。使用这个工具类可以很简洁地编写出优美的多线程程序。ThreadLocal并不是一个Thread,而是Thread的局部变量。当使用ThreadLocal维护变量时,ThreadLocal为每个使用该变量的线程提供独立的变量副本。
ThreadLocal的接口方法:
void set(Object value):设置当前线程的线程局部变量的值。
public Object get():该方法返回当前线程所对应的线程局部变量。
public void remove():将当前线程局部变量的值删除,目的是为了减少内存的占用,该方法是JDK 5.0新增的方法。
需要指出的是,当线程结束后,对应该线程的局部变量将自动被垃圾回收,
所以显式调用该方法清除线程的局部变量并不是必须的操作,但它可以加快内存回收的速度。
protected Object initialValue():返回该线程局部变量的初始值,该方法是一个protected的方法,显然是为了让子类覆盖而设计的。
这个方法是一个延迟调用方法,在线程第1次调用get()或set(Object)时才执行,
并且仅执行1次。ThreadLocal中的缺省实现直接返回一个null。
在JDK5.0中,ThreadLocal已经支持泛型,该类的类名已经变为ThreadLocal<T>。
ThreadLocal是如何做到为每一个线程维护变量的副本的呢?其实实现的思路很简单:在ThreadLocal类中有一个Map,用于存储每一个线程的变量副本
模拟ThreadLocal代码清单:

Java代码 复制代码 收藏代码
  1. package cn.itcast.ref;
  2. import java.util.Collections;
  3. import java.util.HashMap;
  4. import java.util.Map;
  5. import java.util.Random;
  6. public class MyThreadLocal {
  7. //使用同步的map维护对象
  8. private static Map<Thread,Object> map =
  9. Collections.synchronizedMap(new HashMap<Thread,Object>());
  10. public static Object get(){
  11. //获取当前线程
  12. Thread currentThread = Thread.currentThread();
  13. Object o = map.get(currentThread);
  14. if(o==null){
  15. o = new Random().nextInt(100);//假设是一个随机数
  16. map.put(currentThread, o);
  17. }
  18. return o;
  19. }
  20. public static void remove(){
  21. Thread currentThread = Thread.currentThread();
  22. if(map.containsKey(currentThread)){
  23. map.remove(currentThread);
  24. }
  25. }
  26. }