ThreadLocal子线程共享及源码分析

一、发现问题

使用场景: 拦截器赋值(ThreadLocal.set),不依赖上下文传参,同一个线程内传值(ThreadLocal.get)
使用现象: 在父线程ThreadLocal.set,子线程get不到值。

    private final static ThreadLocal<String> threadLocal = new ThreadLocal<>();

    public static void main(String[] args) throws InterruptedException {
        threadLocal.set("MainSet");
        ExecutorService executorService = Executors.newCachedThreadPool();
        executorService.submit(() -> {
            System.out.println(Thread.currentThread().getName() + " get => " + threadLocal.get());
        });
        Thread.sleep(1000);
        executorService.shutdown();
        System.out.println("Main - end");
        
        // out =>
        // pool-1-thread-1 get => null
        // Main - end
    }

二、解决方案

  1. 传参传进子线程。
  2. 使用InheritableThreadLocal。

三、ThreadLocal介绍

作用: 线程隔离、线程独享
看源码: threadLocal.set/threadLocal.get

image.png

看着像threadLocal1.set、threadLocal2.set
实际是Thread.currentThread().ThreadLocalMap.set(ThreadLocal threadLocal1, Object value)、 Thread.currentThread().ThreadLocalMap.set(ThreadLocal threadLocal2, Object value)

image.png

实现线程隔离


image.png

四、为什么InheritableThreadLocal可以实现子线程传递

看源码...InheritableThreadLocal

image.png

由此可见,只是copy,并不共享,copy之后还是隔离
验证....

五、应用场景

  1. 拦截器,获取上下文。
  2. PageHelper。

优点: 减少代码侵入
缺点: 不易维护

六、下期讲解

  1. 内部Map结构、解决hash冲突方式、与jdkHashMap、redis-hash的比较。
  2. 弱引用、弱引用导致的内存泄漏。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容