ThreadLocal 有什么用?
ThreadLocal
类主要解决的就是让每个线程绑定自己的值,可以将ThreadLocal
类形象的比喻成存放数据的盒子,盒子中可以存储每个线程的私有数据。避免了线程安全问题
ThreadLocal 原理
每个Thread
中都具备一个ThreadLocalMap
,而ThreadLocalMap
可以存储以ThreadLocal
为 key ,Object 对象为 value 的键值对。
ThreadLocal 造成内存泄漏的原因?
ThreadLocalMap 中使用的 key 为 ThreadLocal 的弱引用,而 value 是强引用。所以,如果 ThreadLocal 没有被外部强引用的情况下,在垃圾回收的时候,key 会被清理掉,而 value 不会被清理掉。这样一来,ThreadLocalMap 中就会出现 key 为 null 的 Entry。假如我们不做任何措施的
话,value 永远无法被 GC 回收,这个时候就可能会产生内存泄露。
ThreadLocalMap 实现中已经考虑了这种情况,在调用 set()、get()、remove() 方法的时候,会清理掉 key 为 null 的记录。使用完ThreadLocal 方法后 最好手动调用 remove()方法
ThreadLocal 内存泄漏解决方案?
• 每次使用完 ThreadLocal,都调用它的 remove()方法,清除数据。
• 在使用线程池的情况下,没有及时清理 ThreadLocal,不仅是内存泄漏的问题,更严重的是可能导致业务逻辑出现问题。所以,使用 ThreadLocal就跟加锁完要解锁一样,用完就清理
弱引用介绍:
如果对象被弱引用,不管堆内存空间满没满,垃圾回收都会回收掉这个弱引用对象
public class Demo3 {
public static void main(String[] args) throws InterruptedException {
WeakReference<byte[]> reference = new WeakReference<>(new byte[1024 * 1024 * 10]);
System.out.println(reference.get());
System.gc();
System.out.println(reference.get());
}
}
[B@511baa65
null
强引用
如果对象被强引用(就是我们理解的=),那么垃圾回收不会回收掉这个对象
public class Demo1 {
public static void main(String[] args) throws InterruptedException {
ArrayList<String> list = new ArrayList<>();
Collections.addAll(list, "java1", "java2");
System.gc();
System.out.println(list);
}
}
[java1, java2]