简介:本文以一个简要的代码示例介绍ThreadLocal类的基本使用,在此基础上结合图片阐述它的内部工作原理,最后分析了ThreadLocal的内存泄露问题以及解决方法。

欢迎探讨,如有错误敬请指正

如需转载,请注明出处 http://www.cnblogs.com/nullzx/


1. ThreadLocal<T> 简介和使用示例

ThreadLocal只有一个无参的构造方法

  1. public ThreadLocal()

ThreadLocal的相关方法

  1. public T get()
  2. public void set(T value)
  3. public void remove()
  4. protected T initialValue()

initialValue方法的访问修饰符是protected,该方法为第一次调用get方法提供一个初始值。默认情况下,第一次调用get方法返回值null。在使用时,我们一般会复写ThreadLocal的initialValue方法,使第一次调用get方法时返回一个我们设定的初始值。

下面是一个ThreadLocal的一个简单使用示例

  1. package javalearning;
  2.  
  3. import java.util.Random;
  4. import java.util.concurrent.ExecutorService;
  5. import java.util.concurrent.Executors;
  6. import java.util.concurrent.Semaphore;
  7.  
  8. public class ThreadLocalDemo {
  9. /*定义了1个ThreadLocal<Integer>对象,
  10. *并复写它的initialValue方法,初始值是3*/
  11. private ThreadLocal<Integer> tlA = new ThreadLocal<Integer>(){
  12. protected Integer initialValue(){
  13. return 3;
  14. }
  15. };
  16.  
  17. /*
  18. private ThreadLocal<Integer> tlB = new ThreadLocal<Integer>(){
  19. protected Integer initialValue(){
  20. return 5;
  21. }
  22. };
  23. */
  24.  
  25. /*设置一个信号量,许可数为1,让三个线程顺序执行*/
  26. Semaphore semaphore = new Semaphore(1);
  27.  
  28. private Random rnd = new Random();
  29.  
  30. /*Worker定义为内部类实现了Runnable接口,tlA定义在外部类中,
  31. 每个线程中调用这个对象的get方法,再调用一个set方法设置一个随机值*/
  32. public class Worker implements Runnable{
  33. @Override
  34. public void run(){
  35.  
  36. try {
  37. Thread.sleep(rnd.nextInt(1000)); /*随机延时1s以内的时间*/
  38. semaphore.acquire();/*获取许可*/
  39. } catch (InterruptedException e) {
  40. e.printStackTrace();
  41. }
  42.  
  43. int valA = tlA.get();
  44. System.out.println(Thread.currentThread().getName() +" tlA initial val : "+ valA);
  45. valA = rnd.nextInt();
  46. tlA.set(valA);
  47. System.out.println(Thread.currentThread().getName() +" tlA new val: "+ valA);
  48.  
  49. /*
  50. int valB = tlB.get();
  51. System.out.println(Thread.currentThread().getName() +" tlB initial val : "+ valB);
  52. valB = rnd.nextInt();
  53. tlA.set(valB);
  54. System.out.println(Thread.currentThread().getName() +" tlB 2 new val: "+ valB);
  55. */
  56.  
  57. semaphore.release();
  58.  
  59. /*在线程池中,当线程退出之前一定要记得调用remove方法,因为在线程池中的线程对象是循环使用的*/
  60. tlA.remove();
  61. /*tlB.remove();*/
  62. }
  63. }
  64.  
  65. /*创建三个线程,每个线程都会对ThreadLocal对象tlA进行操作*/
  66. public static void main(String[] args){
  67. ExecutorService es = Executors.newFixedThreadPool(3);
  68. ThreadLocalDemo tld = new ThreadLocalDemo();
  69. es.execute(tld.new Worker());
  70. es.execute(tld.new Worker());
  71. es.execute(tld.new Worker());
  72. es.shutdown();
  73. }
  74. }

运行结果

  1. pool-1-thread-1 tlA initial val : 3
  2. pool-1-thread-1 tlA new val: -1288455998
  3. pool-1-thread-3 tlA initial val : 3
  4. pool-1-thread-3 tlA new val: 112537197
  5. pool-1-thread-2 tlA initial val : 3
  6. pool-1-thread-2 tlA new val: -12271334

从运行结果可以看出,每个线程第一次调用TheadLocal对象的get方法时都得到初始值3,注意我们上面的代码是让三个线程顺序执行,显然从运行结果看,pool-1-thread-1线程结束后设置的新值,对pool-1-thread-3线程是没有影响的,pool-1-thread-3线程完成后设置的新值对pool-1-thread-2线程也没有影响。这就仿佛把ThreadLocal对象当做每个线程内部的对象一样,但实际上tlA对象是个外部类对象,内部类Worker访问到的是同一个tlA对象,也就是说是被各个线程共享的。这是如何做到的呢?我们现在就来看看ThreadLocal对象的内部原理。

2. ThreadLocal<T>的原理

首先,在Thread类中定义了一个threadLocals,它是ThreadLocal.ThreadLocalMap对象的引用,默认值是null。ThreadLocal.ThreadLocalMap对象表示了一个以开放地址形式的散列表。当我们在线程的run方法中第一次调用ThreadLocal对象的get方法时,会为当前线程创建一个ThreadLocalMap对象。也就是每个线程都各自有一张独立的散列表,以ThreadLocal对象作为散列表的key,set方法中的值作为value(第一次调用get方法时,以initialValue方法的返回值作为value)。显然我们可以定义多个ThreadLocal对象,而我们一般将ThreadLocal对象定义为static类型或者外部类中。上面所表达的意思就是,相同的key在不同的散列表中的值必然是独立的,每个线程都是在各自的散列表中执行操作

TheadLocal中的get源代码

  1. public T get() {
  2. Thread t = Thread.currentThread();
  3. ThreadLocalMap map = getMap(t);
  4. if (map != null) {
  5. ThreadLocalMap.Entry e = map.getEntry(this);//这里的this是指当前的ThreadLocal对象
  6. if (e != null) {
  7. @SuppressWarnings("unchecked")
  8. T result = (T)e.value;
  9. return result;
  10. }
  11. }
  12. return setInitialValue();
  13. }

3. 由ThreadLocal造成的内存泄露和相应解决办法

ThreadLocalMap 中用内部静态类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类继承了WeakRefrence类,所以一个条目就是一个弱引用类型的对象(要搞清楚,持有weakRefrence对象的引用是个强引用),那么这个weakRefrence对象保存了谁的弱引用呢?我们看到构造函数中有个supe(k),k是ThreadLocal类型对象,super表示是调用父类的构造函数(父类是谁你要想清楚哦?),所以说一个entry对象中存储了ThreadLocal对象的弱引用和这个ThreadLocal对应的value对象的强引用。有关弱引用的相关内容请参考我的另一篇博客《Java中的四种引用以及ReferenceQueue和WeakHashMap的使用示例》

我们现在假设一种情况,假设我们在线程的run方法中调用了一个方法,并在这个方法中创建了ThreadLocal对象,并使用了他,内存结构示意图如下。

当这个方法结束时,这个方法中创建的ThreadLocal对象本身(图中绿色区域)就被垃圾回收器回收了,但是线程还没有结束,所以ThreadLocalMap中还存在这个entry。由于entry中的key(即ThreadLocal对象)是弱引用类型,所以此时调用entry.get()方法时就会返回null,内部结构如下图所示。

从图中我们可以看到value对象(红色区域)始终不能被回收,而我们再也不会使用它了,这就造成了内存泄露。

那Entry中为什么保存的是key的弱引用呢?其实这是为了最大程度上减少内存泄露,副作用是同时减少哈希表中的冲突。当ThreadLocal对象被回收时,对应entry中的key就自动变成null(entry对象本身不为null)。若此后我们调用get,set或remove方法时,就会尝试删除key为null的entry,以释放value对象所占用的内存。

我们现在来看看get方法(上面有get方法的源代码)中调用的getEntry方法。

  1. private Entry getEntry(ThreadLocal<?> key) {
  2. int i = key.threadLocalHashCode & (table.length - 1);
  3. Entry e = table[i];
  4. if (e != null && e.get() == key)
  5. return e;
  6. else
  7. return getEntryAfterMiss(key, i, e);
  8. }

从源代码中我们可以看出,有可能会调用getEntryAfterMiss方法,而在这个方法中,删除key为null的Entry对象。同理set方法也有类似的行为,而remove方法不仅仅删除掉参数ThreadLocal对象对应的entry,而且也会尝试删除其它key为null的entry。

  1. private Entry getEntryAfterMiss(ThreadLocal<?> key, int i, Entry e) {
  2. Entry[] tab = table;
  3. int len = tab.length;
  4.  
  5. while (e != null) {
  6. ThreadLocal<?> k = e.get();
  7. if (k == key)
  8. return e;
  9. if (k == null)
  10. expungeStaleEntry(i);
  11. else
  12. i = nextIndex(i, len);
  13. e = tab[i];
  14. }
  15. return null;
  16. }

但是上述的方式并不能完全解决内存泄露问题,因为我们在这个方法结束的时候逻辑上不一定必须调用get方法,而get方法也不一定执行getEntryAfterMiss方法。所以类本身是没有这个能力的,我们只能在不再使用某个ThreadLocal对象后,手动调用remoev方法来删除它,各自线程中调用共享的ThreadLocal对象的remove方法,这对其它线程是没有影响的,这个应该不难理解。在线程池中这就操作是必须的,不仅仅是内存泄露的问题。因为线程池中的线程是重复使用的,意味着这个线程的ThreadLocalMap对象也是重复使用的,如果我们不手动调用remove方法,那么后面的线程就有可能获取到上个线程遗留下来的value值,造成bug。

4. 参考内容

[1]. Java并发编程:深入剖析ThreadLocal

[2]. [Java并发包学习七]解密ThreadLocal

[3]. 深入分析 ThreadLocal 内存泄漏问题

ThreadLocal原理及使用示例的更多相关文章

  1. 简析ThreadLocal原理及应用

    简析ThreadLocal原理及应用 原创: 东晨雨 JAVA万维猿圈 4月17日 ThreadLocal的源码加上注释不超过八百行,源码结构清晰,代码也比较简洁.ThreadLocal可以说是Jav ...

  2. ThreadLocal原理及其实际应用

    前言 java猿在面试中,经常会被问到1个问题: java实现同步有哪几种方式? 大家一般都会回答使用synchronized, 那么还有其他方式吗? 答案是肯定的, 另外一种方式也就是本文要说的Th ...

  3. java基础解析系列(七)---ThreadLocal原理分析

    java基础解析系列(七)---ThreadLocal原理分析 目录 java基础解析系列(一)---String.StringBuffer.StringBuilder java基础解析系列(二)-- ...

  4. Optaplanner规划引擎的工作原理及简单示例(2)

    开篇 在前面一篇关于规划引擎Optapalnner的文章里(Optaplanner规划引擎的工作原理及简单示例(1)),老农介绍了应用Optaplanner过程中需要掌握的一些基本概念,这些概念有且于 ...

  5. ThreadLocal原理简单刨析

    ThreadLocal原理简单刨析 ThreadLocal实现了各个线程的数据隔离,要知道数据是如何隔离的,就要从源代码分析. ThreadLocal原理 需要提前说明的是:ThreadLocal只是 ...

  6. ScheduleThreadPoolExecutor的工作原理与使用示例

    欢迎探讨,如有错误敬请指正 如需转载,请注明出处 http://www.cnblogs.com/nullzx/ 1. ScheduleExecutorService接口.ScheduledFuture ...

  7. Java并发包中CountDownLatch的工作原理、使用示例

    1. CountDownLatch的介绍 CountDownLatch是一个同步工具,它主要用线程执行之间的协作.CountDownLatch 的作用和 Thread.join() 方法类似,让一些线 ...

  8. Java并发包中CyclicBarrier的工作原理、使用示例

    1. CyclicBarrier的介绍与源码分析 CyclicBarrier 的字面意思是可循环(Cyclic)使用的屏障(Barrier).它要做的事情是,让一组线程到达一个屏障(也可以叫同步点)时 ...

  9. ThreadLocal原理与模拟

    首先用一个程序模拟一下ThreadLocal: public class ThreadLocal1 { private static Dictionary<Thread, Integer> ...

随机推荐

  1. 实现Echarts折线图的虚实转换

    需求:医院的体温单,在统计体温时,对于正常情况下统计的体温数据,需要显示实线:对于进行物理降温后统计的体温数据,需要显示虚线. 现有的体温单是运用 Echarts 折线图,统一用实线显示.因此在这基础 ...

  2. How to sort the dictionary by the value field

    // Sort dictionary by the value field List<KeyValuePair<int, int>> redBallsList = redBal ...

  3. 关于android appcompatv7 Menu items should specify a title的解决办法

    做安卓开发时,添加menu时 是AS报以下错误: 解决办法为修改如下: <menu xmlns:android="http://schemas.android.com/apk/res/ ...

  4. Java 中判断 JSONObject 对应的 VALUE 为空

    目前发现有两种包.两种不一样的json包. 第一种情况是: json包是json-lib包是net.sf.json 怎样判断JSONObject返回的是字符串null还是null值. 研究源码发现.J ...

  5. 【Linux】新建用户 用户组

      案例 hadoop #添加用户组 sudo useradd -s /bin/bash -g hadoop -d /home/hadoop -m hadoop #添加用户 sudo passwd h ...

  6. mysql 数据库选定 创建 删除 变更

    use db_name select * from db_name.tbl_name 显示所有数据库 mysql> select schema_name from information_sch ...

  7. Linux 命令大全之Red Hat 7常用命令总结二

    Linux 命令大全之RedHat7常用命令笔记... ----------------------------------------------------- 征服Linux从终端开始 ----- ...

  8. Andrew Ng机器学习课程笔记--week2(多元线性回归&正规公式)

    1. 内容概要 Multivariate Linear Regression(多元线性回归) 多元特征 多元变量的梯度下降 特征缩放 Computing Parameters Analytically ...

  9. 一个利用pojo类从前端页面request中获取参数的小框架~

    写之前不知道Spring已经实现这样的功能,所以傻傻的写了这个东西! 实现原理挺有趣的在此记录一下.从去年十月参加java开发以来自己终于有了点小进步. 好开心. 解决问题(详解):前端form表单提 ...

  10. 线性代数-矩阵-【4】点乘 C和C++的实现

    点击这里可以跳转至 [1]矩阵汇总:http://www.cnblogs.com/HongYi-Liang/p/7287369.html [2]矩阵生成:http://www.cnblogs.com/ ...