本文主要是介绍JUC并发编程-ThreadLocal,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
1、简介
ThreadLocal 是 Java 中一个重要的工具类,主要用于在多线程环境中提供线程局部变量。这意味着,每个线程都可以拥有自己的变量副本,而不会与其他线程共享这些变量,从而实现线程间的数据隔离。
总结:
- 线程局部变量:ThreadLocal 中填充的变量属于当前线程,对其他线程而言是隔离的。
- 副本独立:每个线程都有自己的实例副本,且只能由当前线程使用。
- 变量隔离:由于每个线程都有独立的实例副本,因此不存在多线程间共享的问题。
2、ThreadLocal 应用场景
- 线程间数据隔离,例如数据库连接管理。
- 线程内数据传递,例如方法间传递参数。
3、ThreadLocal 内存泄露问题
必须回收自定义的ThreadLocal变量,尤其在多线程的场景下,线程会经常被复用,如果不清理自定义的ThreadLocal变量,可能会影响后续业务逻辑和造成内存泄露的问题。
解决方式:
- 尽量使用try-finally块进行回收
4、ThreadLocal 与 四大引用
强引用(Strong Reference)
- 定义:最常见的引用类型,如
Object obj = new Object();
中的obj
就是一个强引用。 - 特点:只要强引用存在,垃圾回收器就不会回收被引用的对象,即使内存不足也不会回收。
- 使用场景:程序中的大多数引用都是强引用。
软引用(Soft Reference)
- 定义:通过
SoftReference
类实现,如SoftReference<Object> softRef = new SoftReference<>(new Object());
。 - 特点:当内存足够时,软引用对象不会被回收;当内存不足时,垃圾回收器会回收软引用对象。
- 使用场景:缓存应用,如网页缓存、图片缓存等。
弱引用(Weak Reference)
- 定义:通过
WeakReference
类实现,如WeakReference<Object> weakRef = new WeakReference<>(new Object());
。 - 特点:弱引用对象更容易被垃圾回收器回收,只要发生垃圾回收,不管内存是否足够,弱引用对象都会被回收。
- 使用场景:ThreadLocal 的 Key,防止内存泄漏。
虚引用(Phantom Reference)
- 定义:通过
PhantomReference
类实现,如PhantomReference<Object> phantomRef = new PhantomReference<>(new Object(), queue);
。 - 特点:虚引用对象任何时候都可能被垃圾回收器回收,必须与引用队列(ReferenceQueue)联合使用,回收前会被加入引用队列。
- 使用场景:用于监控对象的回收,以便在对象被回收时进行一些后续操作。
总结:
只有强引用不会被垃圾回收、软引用只有在内存不足时被垃圾回收、弱引用和虚引用都会被垃圾自动回收
四种引用类型的简单示例:
// 强引用
Object strongRef = new Object();// 软引用
SoftReference<Object> softRef = new SoftReference<>(new Object());// 弱引用
WeakReference<Object> weakRef = new WeakReference<>(new Object());// 虚引用
ReferenceQueue<Object> queue = new ReferenceQueue<>();
PhantomReference<Object> phantomRef = new PhantomReference<>(new Object(), queue);
ThreadLocal内部ThreadLocalMap使用弱引用作为Key:
-
弱引用:
ThreadLocal
的内部实现ThreadLocalMap
使用弱引用作为 Key,指向ThreadLocal
对象。这允许ThreadLocal
对象在没有其他强引用的情况下被垃圾回收,但不会影响ThreadLocalMap
中的值。
五、Thread最佳实践
(1)使用 ThreadLocal.withInitial()
方法来初始化 ThreadLocal
,这样可以确保每个线程在第一次访问时都有一个默认值。
(2)尽量使用static修饰 ThreadLocal
(3)手动调用remove()
这篇关于JUC并发编程-ThreadLocal的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!