ThreadLocal介绍
ThreadLocal叫做线程变量,用于解决多线程并发时访问共享变量的问题。意思是ThreadLocal中填充的变量属于当前线程,该变量对其他线程而言是隔离的,也就是说该变量是当前线程独有的变量。ThreadLocal为变量在每个线程中都创建了一个副本,那么每个线程可以访问自己内部的副本变量。
所谓的共享变量指的是在堆中的实例、静态属性和数组;对于共享数据的访问受Java的内存模型(JMM)的控制,其模型如下
ThreadLoal 变量,线程局部变量,同一个 ThreadLocal 所包含的对象,在不同的 Thread 中有不同的副本。这里有几点需要注意:
因为每个 Thread 内有自己的实例副本,且该副本只能由当前 Thread 使用。这是也是 ThreadLocal 命名的由来。
既然每个 Thread 有自己的实例副本,且其它 Thread 不可访问,那就不存在多线程间共享的问题。
ThreadLocal 提供了线程本地的实例。它与普通变量的区别在于,每个使用该变量的线程都会初始化一个完全独立的实例副本。ThreadLocal 变量通常被private static修饰。当一个线程结束时,它所使用的所有 ThreadLocal 相对的实例副本都可被回收。
总的来说,ThreadLocal 适用于每个线程需要自己独立的实例且该实例需要在多个方法中被使用,也即变量在线程间隔离而在方法或类间共享的场景
下图可以增强理解:
ThreadLocal的简单使用
public class Example {
// 创建一个 ThreadLocal 变量
private static final ThreadLocal<Integer> threadLocal = new ThreadLocal<>();
public static void main(String[] args) {
// 创建两个线程
Thread thread1 = new Thread(() -> {
// 在线程1中设置 ThreadLocal 变量的值
threadLocal.set(1);
System.out.println(Thread.currentThread().getName() + " value: " + threadLocal.get());
});
Thread thread2 = new Thread(() -> {
// 在线程2中设置 ThreadLocal 变量的值
threadLocal.set(2);
System.out.println(Thread.currentThread().getName() + " value: " + threadLocal.get());
});
// 启动线程
thread1.start();
thread2.start();
}
}
Threadlocal 内存溢出问题
ThreadLocal在Java中广泛用于实现线程本地存储,为每个线程提供一个独立的变量副本。然而,不正确的使用可能会导致内存溢出或其他问题。以下是ThreadLocal可能导致内存溢出的原因及解决方法:
内存泄露原因:ThreadLocalMap是ThreadLocal的内部类,用于存储每个线程的变量副本。在ThreadLocalMap中,key是弱引用,而value是强引用。这意味着当没有外部强引用指向ThreadLocal对象时,它可能会被垃圾回收。但是,由于ThreadLocalMap中的key是弱引用,它们在GC时可能会被回收。如果此时ThreadLocal对象仍然没有被回收,那么ThreadLocalMap中就会出现key为null的Entry。
使用static的ThreadLocal:静态的ThreadLocal会延长其生命周期,可能导致内存泄漏。因为静态的ThreadLocal对象在整个应用程序运行期间都存在,如果不正确地使用它,可能会导致内存泄漏。
未调用get(), set(), remove()方法:如果分配了ThreadLocal对象但不再调用其get(), set(), remove()方法,那么ThreadLocal对象可能不会被回收,从而导致内存泄漏。
解决策略:
为避免内存泄漏,应确保对ThreadLocal的使用后调用remove()方法来清除对应的值和引用。
使用弱引用的key可以帮助ThreadLocal自动清理过期的节点,从而避免内存泄漏。
尽量避免使用静态的ThreadLocal,或者确保在使用后正确清理资源。
总之,要正确使用ThreadLocal并避免内存泄漏,需要了解其内部机制和工作原理,并确保在使用时遵循最佳实践。