ThreadLocal底层原理及数据结构详解

  ThreadLocal允许为每个线程创建独立的变量副本,使得同一个ThreadLocal对象在不同的线程中拥有不同的值。它的主要作用是在并发环境下提供线程隔离,避免多个线程共享同一个变量,从而减少线程间的相互干扰。

  ThreadLocal的核心在于为每个线程维护一个独立的数据副本,它的实现主要依赖于每个线程维护一个ThreadLocalMap,这是ThreadLocal专用的Map,用来存储线程自己的变量。

1.1. ThreadLocalMap底层数据结构

  ThreadLocalMap是一个定制化的Map,其结构类似于HashMap,都是以Key-Value的键值对形式进行存储,其中Key存储的是ThreadLocal实例Value存储的是对应的对象,默认为Object。相比于HashMap有一些不同之处:

  • 弱引用的键:ThreadLocalMap的键(即ThreadLocal对象)使用了弱引用(强引用>软引用>弱应用>虚引用),因此当没有其他地方引用该ThreadLocal对象时,GC就会回收它。
  • 线性探测解决哈希冲突:区别于HashMap中的链地址法解决哈希冲突,ThreadLocalMap使用线性探测法来解决哈希冲突,并且负载因子为2/3。
  • 潜在内存泄漏:由于ThreadLocalMap中的键是弱引用,但其存储的Value是强应用,如果ThreadLocal对象被GC回收,而没有调用remove()方法清理值,那么ThreadLocalMap中的值就有可能会一直存在,导致内存泄漏。因此在不适用ThreadLocal后,要及时的调用remove()方法,手动清除线程的副本变量。或者使用 try-finally 模式来保证在完成工作后调用 remove()
1.2. 能否使用ThreadLocal往线程中存储多个副本变量?

         默认情况下,ThreadLocal每个线程只能存储一个值,因为它的设计初衷就是让每个线程独立的维护一组与ThreadLocal对象相关的值,也就是说,每个ThreadLocal实例只能存储一个值。

        虽然 ThreadLocal 本身每个实例只能存储一个值,但多个 ThreadLocal 实例在同一个线程中是存储在 ThreadLocalMap 里的。因此,当一个线程中存在多个 ThreadLocal 实例时,这些实例及其对应的值就会存储在该线程的 ThreadLocalMap 中。

        那如果我们就是想让一个线程拥有多个副本变量该怎么办?

  • 法一:使用ThreadLocal存储一个容器(如Map或自定义对象)

        虽然每个ThreadLocal实例只能存储一个值,但是其存储的是什么值是由我们决定的,因此可以将想要存储的多个变量放入Map中,以此实现存储多个独立的副本变量。

  • 法二:使用多个ThreadLocal对象
private static ThreadLocal<String> threadLocal1 = new ThreadLocal<>();
private static ThreadLocal<Integer> threadLocal2 = new ThreadLocal<>();
threadLocal1.set("Thread1");
threadLocal2.set("Thread2");

        通过以上代码在每个线程的ThreadLocalMap中创建了两个ThreadLocal对象,分别存储"Thread1"和"Thread2",因此可以通过不同的ThreadLocal实例对象来获取不同的值。

ThreadLocalJava中的一个线程局部变量,它提供了一种在多线程环境下保持变量的副本独立性的机制。每个线程都可以独立地修改自己的副本,而不会影响其他线程的副本。 ThreadLocal底层原理主要涉及到两个关键点:ThreadLocalMap和Thread。 1. ThreadLocalMap:每个Thread对象内部都有一个ThreadLocalMap对象,用于存储线程局部变量的副本。ThreadLocalMap是一个自定义的哈希表,它的键是ThreadLocal对象,值是对应线程的变量副本。每个线程可以通过ThreadLocal对象获取自己的变量副本。 2. Thread:每个线程在创建时都会初始化一个ThreadLocalMap对象,并且在整个线程生命周期内都持有该对象。当线程需要获取或设置ThreadLocal变量时,会通过ThreadLocal对象在ThreadLocalMap中进行查找或修改。 具体的操作流程如下: - 当调用ThreadLocal的set方法时,会先获取当前线程的ThreadLocalMap对象,然后将ThreadLocal对象作为键,要设置的值作为值,存储到ThreadLocalMap中。 - 当调用ThreadLocal的get方法时,同样会先获取当前线程的ThreadLocalMap对象,然后根据ThreadLocal对象在ThreadLocalMap中查找对应的值并返回。 - 当线程结束时,由于ThreadLocalMap是存储在Thread对象中的,所以Thread对象会被垃圾回收,从而自动清理掉对应的ThreadLocalMap。 总结一下,ThreadLocal通过在每个线程内部维护一个ThreadLocalMap来实现线程局部变量的存储和访问。每个线程都拥有自己独立的变量副本,互不干扰。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值