Java中的ThreadLocal、ThreadLocalMap原理学习

本文详细探讨了Java中的ThreadLocal和ThreadLocalMap的工作原理。通过实例展示了ThreadLocal如何在不同线程间保持独立的变量副本,并解释了ThreadLocalMap内部结构,强调每个线程拥有自己的threadLocalMap,其key为ThreadLocal对象。同时,文章提到了线程局部变量可能导致的内存泄漏问题,建议在使用完ThreadLocal后调用remove方法以防止内存泄漏。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

public class ThreadLocalTest {
    public static void main(String[] args) {
        final ThreadLocal<Integer> local = new ThreadLocal<>();
        local.set(10);
        Thread t = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName() + " local: " + local.get());
            }
        });
        t.start();
        System.out.println("Main local: " + local.get());
    }
}

输出结果:

Thread-0 local: null

Main local: 10

 

原因是在主线程中set了值,在子线程中没有set。threadLocal保存的是每个线程的值。

解析:

threadLocal类的set方法:

 

key为当前threadLocal,value为要设置的值。

如果为空时,getMap方法:

 

t为当前线程,t.threadLocals的意思是,在每个线程的内部,都有一个threadLocals,这个threadLocals是一个threadLocalMap,源码:

 

也就是说,每个线程内部都有一个threadLocalMap,这个map没有继承Map,而是一个ThreadLocal的内部类,自己实现的map,key规定了只能是threadLocal。具体关于threadLocalMap的解释:threadLocalMap

为什么key为threadLocal呢?

因为每个线程都有一个threadLocalMap,里面可以存放多个threadLocal,这样就可以区分每个线程各自存放的值。

(这里起初很容易误解成多个线程共享同一个threadLocalMap,key为线程,这样的话每个线程只能存放一个值,实际不是这样的。)

一个threadLocal对于一个线程只能存放一个值,如果要一个线程要存放多个值就要创建多个threadLocal。

 

get方法:

 

也是用当前的threadLocal为key来get值的。源码中的getMap方法就是拿的当前thread的threadLocals。

 

ps:threadLocalMap中的key为弱引用,在下一次GC时会被回收掉,但是value是强引用,是不会回收掉的,这里有发生内存泄漏的可能。最好的是在get完之后调用remove,断开引用链,这样value就可以被gc回收掉了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值