Java中的ThreadLocal

一、概念

  • ThreadLocal可以实现资源对象的线程隔离,让每个线程各用各的资源对象,避免争用引发的线程安全问题。
  • ThreadLocal同时实现了线程内的资源共享。

二、原理

  • ThreadLocal本质上是依托于每个线程内的ThreadLocalMap类型的成员变量,该成员变量是用来存储资源对象的。
    1. 调用set方法,就是以ThreadLocal自己作为key,资源对象作为value,放入当前线程的ThreadLocalMap集合中。
    2. 调用get方法,就是以ThreadLocal自己作为key,到当前线程中查找关联的资源值。
    3. 调用remove方法,就是以ThreadLocal自己作为key,移除当前线程关联的资源值。
  • 每创建一个ThreadLocal对象,就会计算出一个该对象的哈希值,此值就用于计算该ThreadLocal对象应该存储在每个线程的ThreadLocalMap成员变量的哪个桶位上。
    1. 第一个ThreadLocal的哈希值为0。
    2. 随后,每创建一个ThreadLocal对象,就会计算一个哈希值,且新的哈希值是在上一个ThreadLocal对象哈希值的基础上增加了1640531527。
    3. 当ThreadLocalMap成员变量的存储数目达到了当前容量的三分之二时,就会触发扩容,将容量扩容为当前的二倍。
    4. 如果存储ThreadLocal对象时,在ThreadLocalMap成员变量中发生了哈希冲突,则会寻找当前ThreadLocalMap成员变量中发生哈希冲突桶位的下一个空闲桶位存放该对象和资源(开放寻址法)。

三、细节

1.为什么ThreadLocalMap中的key(即ThreadLocal)要设计为弱引用?

  • Thread可能需要长时间运行(如线程池中的线程),如果key不再使用,需要在内存不足时进行垃圾回收(GC)来释放其占用的内存。注意,只有设置成弱引用才可以在GC时被释放,强引用无法释放。

2.何时释放ThreadLocalMap中的value(即ThreadLocal内资源)的内存?

  • GC仅是让key的内存释放,后续值的内存的释放,是要根据key是否为null来进行的,有如下几个释放时机
    1. get key发现null key。注意,此时ThredLocal还会把当前查询的key重新放入,只不过值是null。
    2. set key时,会使用启发式扫描(临近清除),清除临近的null key,启发次数与元素个数、是否发现null key有关。
    3. 前两种情况都是基于key(也就是ThreadLocal对象)为null的情况,但是在实际使用中,ThreadLocal创建时都把它作为静态变量,是强引用,GC无法回收,所以我们都使用remove方法操作ThreadLocalMap成员变量将其清理,此时也会释放key和value的内存。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
ThreadLocalJava的一个类,用于在多线程环境下实现线程本地变量。它提供了一种线程级别的数据隔离机制,使得每个线程都可以拥有自己独立的变量副本,互不干扰。每个线程都可以通过ThreadLocal对象来访问自己的变量副本,而不会影响其他线程的副本。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [JavaThreadLocal详解](https://blog.csdn.net/licux/article/details/117292777)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 33.333333333333336%"] - *2* [JavaThreadLocal详解(一篇就够了)](https://blog.csdn.net/qq_38721537/article/details/124565091)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 33.333333333333336%"] - *3* [JavaThreadLocal详解](https://blog.csdn.net/qq_53729147/article/details/127967751)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 33.333333333333336%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值