ThreadLocal是Java中用于管理线程局部变量 的类,它为每个线程提供独立的变量副本,从而解决多线程并发访问共享资源时的线程安全问题。
核心原理与实现
ThreadLocal的实现基于ThreadLocalMap结构,每个线程(Thread类)内部都维护一个ThreadLocal实例为键,变量副本为值的映射表。当线程首次调用ThreadLocal的get()或set()方法时,会初始化该线程的变量副本。ThreadLocal变量声明为private static字段,这样类的所有实例都可以访问它,但每个线程操作的都是自己的副本。
主要方法
ThreadLocal类提供了四个核心方法
- set(T value):设置当前线程的变量副本值。
- T get():获取当前线程的变量副本值。
- remove():移除当前线程的变量副本。
- protected T initialValue():返回该线程局部变量的初始值,默认实现返回null,可以通过重写此方法来提供自定义初始值。
典型应用场景
- 线程上下文传递:在跨线程调用场景中,可以使用ThreadLocal在不修改方法签名的情况下传递线程上下文信息,如用户信息或请求ID。
- 数据库连接管理:在使用数据库连接池的情况下,将数据库连接存储在ThreadLocal中,使每个线程能独立管理自己的连接,例如MyBatis中的SqlSession对象就使用了ThreadLocal。
- 事务管理:在需要手动管理事务的场景中,使用ThreadLocal存储事务上下文,保证事务的隔离性。
注意事项
使用ThreadLocal时需要注意内存管理,应在不使用时主动调用remove()方法清理变量副本,防止内存泄露。
线程外调用:remove()方法应在当前线程的任务执行完毕(如线程结束或任务完成)后调用,目的是清除当前线程的ThreadLocal变量副本,避免内存泄漏。
每一个thread内有一个threadLocalMap。threadLocalMap的key为threadLocal(弱引用)。
当threadLocal = null时(ThreadLocal对象的强引用被释放后,由于Entry的key是弱引用,这个ThreadLocal对象在GC时会被回收,导致key变为null),值value还在map中。没有被清除。
只要线程持续运行(线程池中的线程),这些value对象就无法被回收,从而造成内存泄漏。