news 2026/3/23 10:21:33

一文看懂ThreadLocal的原理

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
一文看懂ThreadLocal的原理

先看下ThreadLocal的基本用法,创建5个线程给同一个ThreadLocal变量设置不同的值,从打印结果看每个线程设置和获取的值都是不同的,可见ThreadLocal为线程安全的,每个线程保存的值相互独立。

public class ThreadLocalTest { public static ThreadLocal local = new ThreadLocal(); public static void main(String[] args) { for (int i = 0; i < 5; i++){ new Thread(new Runnable() { @Override public void run() { local.set("hello" + Thread.currentThread()); System.out.println(local.get()); } }).start(); } } } //打印结果为: helloThread[Thread-4,5,main] helloThread[Thread-3,5,main] helloThread[Thread-2,5,main] helloThread[Thread-0,5,main] helloThread[Thread-1,5,main]

现在开始ThreadLocal的原理,过程比较复杂。首先看下ThreadLocal的set()方法存数据的过程,获取调用set方法的线程中持有的ThreadLocalMap(ThreadLocal.ThreadLocalMap threadLocals = null;),每个线程的ThreadLocalMap都是独立的,因此存储的值是不同的。

public void set(T value) { Thread t = Thread.currentThread(); ThreadLocalMap map = getMap(t); if (map != null) { map.set(this, value); } else { createMap(t, value); } } // 获取线程内置的ThreadLocalMap ThreadLocalMap getMap(Thread t) { return t.threadLocals; }

如果在一个线程中首次使用ThreadLocal保持数据,则需要创建ThreadLocalMap,ThreadLocalMap中保存数据的实体是Entry,保存数据的过程就是先计算这个ThreadLocal对象的hashcode,根据hashcode计算在Entry数组中的位置,然后将创建的Entry保存在这个位置。

void createMap(Thread t, T firstValue) { t.threadLocals = new ThreadLocalMap(this, firstValue); } ThreadLocalMap(ThreadLocal<?> firstKey, Object firstValue) { table = new Entry[INITIAL_CAPACITY]; int i = firstKey.threadLocalHashCode & (INITIAL_CAPACITY - 1); table[i] = new Entry(firstKey, firstValue); size = 1; setThreshold(INITIAL_CAPACITY); } // 弱引用 static class Entry extends WeakReference<ThreadLocal<?>> { /** The value associated with this ThreadLocal. */ Object value; Entry(ThreadLocal<?> k, Object v) { super(k); value = v; } }

调用set()设置值的时候,会根据ThreadLocal计算hashcode。ThreadLocal中的属性threadLocalHashCodeprivate final int threadLocalHashCode = nextHashCode();用来维护每个ThreadLocal的hash值。再根据hashcode计算Entry数组的索引,根据索引找到当前线程对应的Entry,这里分三种情况:

  1. 第一次设值,则直接添加如果是当前线程使用的ThreadLocalif (k == key),则将对象设置进来,即写到存储数据的Entry中
  2. ThreadLocalif (k == key)已经有值了,就直接更新
  3. ThreadLocal作为临时变量被gc了if (k == null),例如在方法代码块中声明的ThreadLocal临时变量在方法执行完时不存在了,Entry中的ThreadLocal作为key是弱引用就会被gc;或者线程销毁了,此时指向Entry的引用不存在了,Entry也会被gc,此时如果不gc的话就会出现一块无法访问到的Entry,造成内存泄漏。

set()时如果发现hash冲突,ThreadLocal的做法是向后移动一位,到数组的下个索引处保存Entry,如果下个索引处有值了再继续向后找。

private void set(ThreadLocal<?> key, Object value) { Entry[] tab = table; int len = tab.length; int i = key.threadLocalHashCode & (len-1); for (Entry e = tab[i]; e != null; e = tab[i = nextIndex(i, len)]) { ThreadLocal<?> k = e.get(); // 更新 if (k == key) { e.value = value; return; } // 临时ThreadLocal被回收的处理 if (k == null) { replaceStaleEntry(key, value, i); return; } } // 第一次直接设置 tab[i] = new Entry(key, value); int sz = ++size; if (!cleanSomeSlots(i, sz) && sz >= threshold) rehash(); } // 向后移动一位 private static int nextIndex(int i, int len) { return ((i + 1 < len) ? i + 1 : 0); }

当通过get()方法获取数据时,首先找到当前的线程对象,获取线程对象内部的ThreadLocalMap,然后根据ThreadLocal对象计算Entry的索引,找到本线程存储数据的Entry,获取Entry中的数据。

public T get() { Thread t = Thread.currentThread(); ThreadLocalMap map = getMap(t); if (map != null) { ThreadLocalMap.Entry e = map.getEntry(this); if (e != null) { @SuppressWarnings("unchecked") T result = (T)e.value; return result; } } return setInitialValue(); } private Entry getEntry(ThreadLocal<?> key) { int i = key.threadLocalHashCode & (table.length - 1); Entry e = table[i]; if (e != null && e.get() == key) return e; else return getEntryAfterMiss(key, i, e); }
  • ThreadLocal内存泄漏的问题
    可以看到Entry是指向ThreadLocal的弱引用,弱引用不会阻止gc的垃圾回收,如果这个ThreadLocal对象置为null,指向ThreadLocal对象的弱引用不会阻止gc的垃圾回收,此时ThreadLocal对象会被gc回收,通过get()方法获取value时需要计算ThreadLocal对象的hashcode,在ThreadLocal对象被回收的情况就无法计算hashcode,也就无法访问这个value引用的对象,于是value就成了有引用链但是无法被访问的内存,即造成内存泄漏了。
static class Entry extends WeakReference<ThreadLocal<?>> { /** The value associated with this ThreadLocal. */ Object value; Entry(ThreadLocal<?> k, Object v) { super(k); value = v; } }

解决方法:

  1. 将ThreadLocal变量定义成private static,这样就一直存在ThreadLocal的强引用,可以通过ThreadLocal对象访问到保存的数据,不会造成内存泄漏
  2. 调用remove()方法清除内存
public void remove() { ThreadLocalMap m = getMap(Thread.currentThread()); if (m != null) { m.remove(this); } } private void remove(ThreadLocal<?> key) { Entry[] tab = table; int len = tab.length; int i = key.threadLocalHashCode & (len-1); for (Entry e = tab[i]; e != null; e = tab[i = nextIndex(i, len)]) { if (e.get() == key) { e.clear(); expungeStaleEntry(i); return; } } }

总结一下,每个ThreadLocal都有每个线程对应的ThreadLocalMap用于保存数据,每个线程的ThreadLocalMap对象都不相同,所以是线程安全的。ThreadLocal存在内存泄漏问题,需要持有ThreadLocal的强引用或remove清理。有不对的地方请大神指出,欢迎大家一起讨论交流,共同进步,更多请关注微信公众号 葡萄开源。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/16 5:42:05

基于单片机的水缸加热温控控制系统设计

一、系统整体设计方案 本系统以 STC89C52RC 单片机为控制核心&#xff0c;聚焦水产养殖、家庭储水加热等场景的水缸水温管控需求&#xff0c;可实现水温实时采集、目标温度设定、自动加热调节、超温保护及状态反馈功能&#xff0c;兼顾控温精度与使用安全性&#xff0c;为水缸水…

作者头像 李华
网站建设 2026/3/19 3:01:21

铝板加热件工艺设计

摘 要 传送带是生产生活中不可缺少的&#xff0c;它可使我们的工作提高很多效率&#xff0c;但是传送带损坏会给生产带来很多麻烦&#xff0c;但是传送带的损坏又是不可避免的&#xff0c;所以我们需要快速修理传送带的机器——皮带硫化机。 铝板加热件是皮带硫化机中的加热零…

作者头像 李华
网站建设 2026/3/15 19:37:01

基于 8086 双机通信数据互发仿真设计

一、数据互发核心目标与场景设定 在双机通信系统中&#xff0c;数据互发是实现信息交互的核心功能&#xff0c;直接影响系统协同工作的效率与可靠性。基于 8086 的双机串行口通信系统&#xff0c;需确保两台主机能实时、准确地双向传输数据&#xff0c;涵盖单字符即时互发和多字…

作者头像 李华
网站建设 2026/3/18 8:53:13

你的内测用户,正在被糟糕的分发流程劝退

我们痴迷于谈论“用户体验”&#xff08;UX&#xff09;&#xff0c;为App里的每一个按钮、每一个交互细节反复打磨。但我们似乎常常忽略了另一群人的体验——那些在我们产品正式上线前&#xff0c;就愿意投入时间、陪我们一起成长的内-测用户。他们可能是你的朋友、同事&#…

作者头像 李华
网站建设 2026/3/21 11:15:30

Java源码:无人共享健身房物联网方案

以下是一个基于Java的无人共享健身房物联网方案源码级实现框架&#xff0c;包含架构设计、核心模块、技术细节及优化策略&#xff1a; 一、系统架构设计 分层架构&#xff1a;采用Spring Cloud服务架构&#xff0c;实现高可用性和可扩展性。 表现层&#xff1a;使用UniApp实现…

作者头像 李华
网站建设 2026/3/15 23:10:34

Java源码:打造无人共享健身房物联网

以下是一套基于Java的无人共享健身房物联网系统源码实现方案&#xff0c;涵盖系统架构、核心模块、技术细节及优化策略&#xff1a;一、系统架构设计采用分层架构&#xff0c;基于Spring Cloud构建微服务系统&#xff0c;实现高可用性和可扩展性。系统主要分为以下几个层次&…

作者头像 李华