详解ThreadLocal原理及内存泄漏

1. ThreadLocal作用

ThreadLocal的作用是使得每个线程都能拥有各自独立的对象副本,假设多个线程拥有同一个实例,ThreadLocal<T>类型的变量在每个线程中都有一个副本,从而为变量提供了线程间隔离的作用。

2. ThreadLocal实例


public class ThreadLocalDemo implements Runnable{
    private static ThreadLocal<People> local = new ThreadLocal<>();

    public static void main(String[] args) throws InterruptedException {
        ThreadLocalDemo localDemo = new ThreadLocalDemo();
        Thread thread1 = new Thread(localDemo);
        thread1.start();

      	Thread.sleep(1000);
      
        Thread thread2 = new Thread(localDemo);
        thread2.start();
    }

    @Override
    public void run() {
        People people = local.get();
        if (people==null) {
            people = new People();
            people.setName("djh");
            people.setAge(new Random().nextInt());
            local.set(people);
        }
        System.out.println(people);
    }
}

public class People {
    String name;
    int age;

    public void setName(String name) {
        this.name = name;
    }

    public void setAge(int age) {
        this.age = age;
    }

}

在该例子中,thread1thread2拥有同一个对象localDemo,如果不使用ThreadLocal包装,则两个线程访问的people应该是同一个。而在我们使用ThreadLocal包装后,运行结果如下:

bean.People@6fb35e7d
bean.People@614b201a

两者访问的people是不一样的两个对象,而这就是ThreadLocal的作用,提供了变量线程间的隔离能力。

3. 源码解析

查看ThreadLocal的set方法,

public void set(T value) {
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    if (map != null)
        map.set(this, value);
    else
        createMap(t, value);
}
  1. 先获取当前线程;
  2. 获取当前线程的threadLocals变量(是一个ThreadLocalMap类型的Map);
  3. 如果map不为空,为这个map插入<当前ThreadLocal对象,值>的键值对;
  4. 如果该map为空,则创建一个一个ThreadLocalMap对象,t.threadLocals = new ThreadLocalMap(this, firstValue)

由于每个线程都有各自的threadLocals,所以多个线程运行时经过第二步拿到的map是不一样的,这样就实现了线程间隔离。

4. ThreadLocal的内存泄漏问题

由第3节可以得知,数据实际存在每个线程的threadLocals成员(即一个ThreadLocalMap对象)里,ThreadLocalMap和普通的map一样使用Entry数组存储数据,这个Entry特殊的地方在于,它继承了弱引用,即该Entry的key是一个ThreadLocal对象的弱引用。

static class Entry extends WeakReference<ThreadLocal<?>> {
    /** The value associated with this ThreadLocal. */
    Object value;

    Entry(ThreadLocal<?> k, Object v) {
        super(k);
        value = v;
    }
}

弱引用的特点是,如果某个对象只被弱引用关联,在下一次GC时该对象就会被回收。此时某个ThreadLocal类型变量使用完被我们置为null,此时Entry中关联的Key会变成null,但Value由于是强引用,且运行的线程指向他,因此Value部分发生了内存泄漏。

线程执行完毕被回收时Value会随之回收,但当我们使用线程池时,由于线程会复用,因此Value会一直存在,这就发生了更为严重的内存泄漏。

5. 参考资料

  1. 正确理解Thread Local的原理与适用场景
  2. ThreadLocal 内存泄漏问题深入分析
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
ThreadLocalJava 中的一个线程局部变量,它为每个线程提供了独立的变量副本,每个线程都可以通过 ThreadLocal 对象来访问自己的变量副本,而不会影响其他线程的副本。ThreadLocal多线程编程中非常有用,可以解决线程安全问题。 使用 ThreadLocal 的主要场景包括: 1. 线程上下文信息传递:有些情况下,我们需要在多个方法之间传递一些上下文信息,例如用户认证信息、数据库连接等。使用 ThreadLocal 可以避免在方法参数中传递这些上下文信息,每个线程都可以独立地访问自己的上下文信息。 2. 线程安全的对象:有些对象在多线程环境下是不安全的,如果每个线程都持有一个对象的副本,就可以避免多线程竞争访问导致的安全问题。例如 SimpleDateFormat 是非线程安全的,可以使用 ThreadLocal 来为每个线程提供一个独立的 SimpleDateFormat 对象。 3. 隐式参数传递:有些方法需要依赖某些参数,但是这些参数对于调用方来说并不是必须的。使用 ThreadLocal 可以将这些参数设置为 ThreadLocal 变量,在方法中直接获取这些参数,而不需要显式传递。 需要注意的是,使用 ThreadLocal 时要及时清理资源,避免内存泄漏。在使用完毕后,应该调用 ThreadLocal 的 remove() 方法来清理当前线程持有的变量副本。 总之,ThreadLocal 可以用于在多线程环境下实现线程安全、线程间数据隔离和传递上下文信息等功能。但是过度使用 ThreadLocal 也会导致代码可读性变差,增加了代码的复杂性,需要根据具体场景进行合理使用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值