Java 安装 Java:第一个程序 Hello World Java:建议使用 UTF-8 编写 Java 代码 Java:package 包命名规范 使用 Intellij IDEA 创建 Java 项目 Java 布尔类型 Java 处理日期和时间 Java 正则表达式 Java finalize 方法 Java:空值 null Java 如何触发垃圾回收 Java ThreadLocal Java InheritableThreadLocal Java Integer之间的比较 Java 动态代理 Java 匿名类 Java 枚举 Java 如何静态导入 import static println Java 引用级别:强引用、软引用、弱引用、幽灵引用 Java try finally return 解惑 Java WeakHashMap Java ReferenceQueue 怎么写 Java 示例代码? Java 匿名类双大括号初始化 什么是 Java Bean Java 多行字符串 Java 快速生成 List Java 快速生成 Map Java 将异常堆栈转换为 String JDK SPI 的使用和源码分析 Java Map 中的 key 和 value 能否为 null ? Java List 和 数组的互相转换 Java 获取环境变量 Java 获取和设置系统属性 Java:如何获取当前进程的 PID ? Java 字符串左侧 右侧补充空格或者其他字符 Java 线程 Java:如何获取文本文件内容 Java:读取资源文件内容 Java:使用 JavaFx 构建 GUI Java:Class 类 Java:使用 instanceof 判断对象类型 一个自定义的 Java 工具类 Java:获取当前函数所属类的类名 Java:获取当前执行的函数名 Java:使用 String 的 split 函数拆分字符串 Java:获取字符的 Unicode 编号(代码点) Java:获取当前工作目录 Java:使用 Class 对象的 isArray 方法判断对象是否为数组 使用 Java 生成 CSV 文件 Java Mockito 测试框架快速入门 JUnit 入门 JUnit 单测隔离 Java JOOR 反射库 Java alibaba transmittable-thread-local 库:让 ThreadLocal 跨线程传播 Java 日志组件 slf4j 的使用和源码分析 Java Lombok 库:为你减少样板代码 Java:使用 cglib 实现动态代理 Java Hibernate validator 校验框架 Java 使用 Hessian2 序列化和反序列化 H2 数据库快速入门 Java:使用 Gson 库处理 JSON 数据 Java 集成 groovy 构建规则引擎 Java 13:安装 Java 13 新特性:文本块(多行字符串) 卸载 MacOS 上安装的 Java Java:执行 sql 文件 Java JDK 有哪些发行版 ? java拾遗:String和数组 java拾遗:由反转数组想到System.out的实现机制 java拾遗:如何读取properties文件内容 Java并发概念汇总 java拾遗:System.out.println()是什么? java拾遗:通过示例理解位运算 使用“庖丁解牛”进行中文分词 DBUtils简明教程 试用velocity模板引擎 Java:将字符串哈希为数字 kafka SnappyError no native library is found 问题

Java InheritableThreadLocal


#Java


Java InheritableThreadLocal 继承自 ThreadLocal,是 Java ThreadLocal 的升级版。

这个升级体现在,子线程会继承父线程的 InheritableThreadLocal 值。

建议先阅读: Java ThreadLocal

基本使用

public class InheritableThreadLocalTest {

    private static InheritableThreadLocal<Integer> logId = new InheritableThreadLocal<>();

    public static void showLogId() {
        System.out.printf("%s : %s\n", Thread.currentThread().getName(), logId.get());
    }

    public static void main(String[] args) {
        showLogId();

        logId.set(10);
        showLogId();

        logId.set(20);
        showLogId();
    }
}

运行结果:

main : null
main : 10
main : 20

在多线程中使用

下面的 logId、showLogId 来自上面的示例。

showLogId();

logId.set(10);
showLogId();

Thread t1 = new Thread(()->{
    showLogId();
    logId.set(20);
    showLogId();
});

Thread t2 = new Thread(()->{
    showLogId();
    logId.set(30);
    showLogId();
});

t1.start(); // 运行线程 t1
t2.start(); // 运行线程 t2
t1.join();  // 等待线程 t1 执行完
t2.join();  // 等待线程 t2 执行完

showLogId(); // 再看下当前线程的 logId 值

运行结果:

main : null
main : 10
Thread-0 : 10
Thread-0 : 20
Thread-1 : 10
Thread-1 : 30
main : 10

可以看到,main线程在将logId设置为10之后,新建两个线程,在这两个线程(Thread-0 、Thread-1)中 logId 的初始值都是10。

在线程池中使用依然可能遇到问题

如果线程池中线程会被复用,这时 InheritableThreadLocal 的值也会被复用。

为什么说依然,因为 ThreadLocal 也会遇到这种情况。见 Java ThreadLocal

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;

public class InheritableThreadLocalTest {

    private static InheritableThreadLocal<Integer> logId = new InheritableThreadLocal<>();

    public static void showLogId() {
        System.out.printf("%s : %s\n", Thread.currentThread().getName(), logId.get());
    }

    public static void main(String[] args) throws InterruptedException {
        showLogId();

        logId.set(10);
        showLogId();

        // 只有一个线程的线程池,该线程会复用
        ExecutorService executorService = Executors.newFixedThreadPool(1);
        // 提交task
        executorService.submit(() -> {
            showLogId();
            logId.set(20);
            showLogId();
        });

        // 再提交一个task
        executorService.submit(() -> {
            showLogId();
            logId.set(30);
            showLogId();
        });

        // 关闭线程池
        executorService.shutdown();
        // 等待至任务都执行完,最多等3秒
        executorService.awaitTermination(3, TimeUnit.SECONDS);

        showLogId();
    }
}

运行结果:

main : null
main : 10
pool-1-thread-1 : 10
pool-1-thread-1 : 20
pool-1-thread-1 : 20
pool-1-thread-1 : 30
main : 10

线程池只有一个线程处理任务,任务1执行后,logId从10变成20;任务2开始执行时,logId 不是 10,而是20。

不要对内部的值进行修改

public class InheritableThreadLocalTest {

    private static class LogId {
        private Integer value;

        public LogId(Integer logId) {
            this.value = logId;
        }

        public Integer getValue() {
            return value;
        }

        public void setValue(Integer value) {
            this.value = value;
        }

    }
    private static InheritableThreadLocal<LogId> logId = new InheritableThreadLocal<>();


    public static void showLogId() {
        System.out.printf("%s : %s\n", Thread.currentThread().getName(), logId.get().getValue());
    }

    public static void main(String[] args) throws InterruptedException {

        logId.set(new LogId(0));  // 设置LogId,value为0

        logId.get().setValue(10); // 修改了LogId对象的内部值
        showLogId();

        Thread t1 = new Thread(()->{
            showLogId();
            logId.get().setValue(20);  // 修改了LogId对象的内部值
            showLogId();
        });

        t1.start(); // 运行线程 t1
        t1.join();  // 等待线程 t1 执行完

        showLogId(); // 再看下当前线程的 logId value 值
    }
}

运行结果:

main : 10
Thread-0 : 10
Thread-0 : 20
main : 20

上面的示例中,由于两个线程中 InheritableThreadLocal 变量 logId 中的 LogId 对象是同一个,所以子线程修改值为 20 之后, main 线程输出 logId 的value 也变成了 20 。 这个是有问题的,不要用logId.get().setValue(10);,而应该用logId.get(new LogId(10));

InheritableThreadLocal 原理

Java ThreadLocal 介绍了 ThreadLocal 的原理。因为 InheritableThreadLocal 继承自 ThreadLocal,所以原理差不多。

不同点:

  • 每个Thread对象都有一个 ThreadLocalMap 类型的实例变量 threadLocals,也要一个同类型的实例变量 inheritableThreadLocals。InheritableThreadLocal 用 inheritableThreadLocals 存数据。
  • 在线程1中新建线程2时,因为线程2的创建是在线程1中,所以线程2可以拿到线程1的 inheritableThreadLocals ,然后复制一份给自己。于是,线程2启动后,便继承了线程1 inheritableThreadLocals 。


( 本文完 )