ThreadLocal的介绍+经典应用场景

本文深入探讨了Java中的ThreadLocal,解释了其如何解决线程安全问题,与Synchronized的区别,提供了简单的使用示例,并介绍了ThreadLocal在SpringWeb中的应用以及慎用场景。此外,还讲解了ThreadLocal底层源码、弱引用的重要性以及在面试中的相关问题。

🏅作者小三是刚刚毕业不久全栈工程师,写的技术文章基本上是学习过程中笔记整理而来,大家看了之后如果喜欢可以给小弟点点赞哦。
🏅例外小弟还有个程序员交流群,欢迎各位大佬来摸鱼哈。点击加群

什么是ThreadLocal

ThreadLocal又叫做线程局部变量,全称thread local variable,它的使用场合主要是为了解决多线程中因为数据并发产生不一致的问题。ThreadLocal为每一个线程都提供了变量的副本,使得每一个线程在某一时间访问到的并不是同一个对象,这样就隔离了多个线程对数据的数据共享,这样的结果无非是耗费了内存,也大大减少了线程同步所带来的性能消耗,也减少了线程并发控制的复杂度。

总的来说:ThreadLocal适用于每一个线程需要自己独立实例,而且实例的话需要在多个方法里被使用到,也就是变量在线程之间是隔离的但是在方法或者是类里面是共享的场景

那ThreadLocal和Synchronized又有什么区别呢?

虽然ThreadLocal和Synchonized都用于解决多线程的并发访问,但是它们之间还是会有一些本质上的区别的:

Synchronized是利用锁的机制,使得变量或者是代码块在某一时刻里只能被一个线程来进行访问。ThreadLocal是为每一个线程都提供了一个变量的副本,这样就是的每一个线程在某一时刻里访问到的不是同一个对象,这样就隔离了多个线程对数据的数据共享,Synochronized正好相反,可以用于多个线程之间通信能够获得数据共享。

注:ThreadLocal不可以使用原子类型,只能使用Object类型

ThreadLo

### ThreadLocal原理 ThreadLocalJava中用于实现线程隔离的工具,每个使用ThreadLocal的线程都有自己独立的副本。其核心原理与ThreadLocalMap有关,Thread类中有一个类型为ThreadLocalMap的成员变量`threadLocals`,用于存储该线程的所有ThreadLocal变量及其对应的值。 当调用ThreadLocal的`set`方法时,会先获取当前线程的ThreadLocalMap,如果该Map不存在则创建一个,然后以当前ThreadLocal实例作为key,将值存储到Map中。示例代码如下: ```java public class ThreadLocalTest { public static ThreadLocal<Long> x = new ThreadLocal<Long>() { @Override protected Long initialValue() { System.out.println(Thread.currentThread().getId() + " Initial value run..."); return Thread.currentThread().getId(); } }; public static void main(String[] args) { new Thread(() -> { System.out.println(x.get()); }).start(); x.set(100L); System.out.println(x.get()); } } ``` 当调用`get`方法时,同样会先获取当前线程的ThreadLocalMap,然后以当前ThreadLocal实例为key从Map中获取对应的值。若Map中不存在该key,则调用`initialValue`方法初始化一个值并存储到Map中。 ThreadLocalMap的Entry继承自WeakReference,即这个ThreadLocalMap的Key是弱引用,在GC时会被回收。当线程的生命周期大于ThreadLocal的生命周期时,可能存在ThreadLocalMap`<null, Object>`的情况(ThreadLocal被回收,ThreadLocal关联的线程共享变量还存在),这个Object就是泄漏的对象。为避免内存泄漏,应在线程结束前,在`try...finally...`中执行ThreadLocal的`remove`方法[^2]。 ### ThreadLocal应用场景 - **保证线程安全**:在多线程环境中,某些对象不希望被多个线程共享,使用ThreadLocal可以为每个线程创建独立的副本,避免线程安全问题。 - **跨层数据传递**:在复杂的调用链路中,需要在不同层次的方法中传递一些全局参数,使用ThreadLocal可以避免在方法参数中层层传递这些参数,简化代码。 - **复杂调用链路的全局参数传递**:在一些分布式系统中,一次请求可能会经过多个服务,每个服务可能会有多个方法调用。可以使用ThreadLocal在整个调用链路中传递一些全局参数,如请求ID、用户信息等。 - **数据库连接管理**:在多线程环境中,每个线程需要独立的数据库连接,使用ThreadLocal可以为每个线程管理自己的数据库连接,避免连接的混乱和冲突。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值