ThreadLocal是什么

ThreadLocal是线程Thread中属性threadLocals即ThreadLocal.ThreadLocalMap的管理者,ThreadLocal用于给每个线程操作自己线程的本地变量,通过线程私有从而保证线程安全性。

ThreadLocal原理

get()方法来说,线程的本地变量是存放在线程实例的属性ThreadLocalMap上的,ThreadLocalMap本质上就是一个HashMap,ThreadLocal只是一个管理者,当我们的线程需要拿到自己的本地变量时,我们直接调用ThreadLocal去get本地变量即可。

因为get()方法底层会先获取到当前线程,然后通过当前线程拿到他的属性值ThreadLocalMap,如果ThreadLocalMap为空,则会调用ThreadLocal的初始化方法拿到初始值返回,如果不为空,则会拿该ThreadLocal作为key去获取该线程下的ThreadLocalMap里对应的value值。

ThreadLocal内存泄漏问题

线程的属性值ThreadLocalMap中使用的 key 为 ThreadLocal 的弱引用,而value是强引用。所以,如果ThreadLocal没有被外部强引用的情况下,在垃圾回收的时候,key 会被清理掉,而value 不会被清理掉。这样的话,ThreadLocalMap 中就会出现 key 为 null 的 Entry。假如我们不做任何措施的话,value 永远无法被 GC 回收,这个时候就可能会产生内存泄露。

因此针对这种情况,我们有两种原则:

  1. ThreadLocal申明为private static final。JDK建议ThreadLocal定义为private static,这样ThreadLocal的弱引用问题则不存在了。

    • private与final 尽可能不让他人修改变更引用。
    • static 表示为类属性,只有在程序结束才会被回收。
  2. ThreadLocal使用后务必调用remove方法。
    • 最简单有效的方法是使用后将其移除。

关于InheritableThreadLocal

InheritableThreadLocal类是ThreadLocal类的子类。ThreadLocal中每个线程拥有它自己的值,与ThreadLocal不同的是,InheritableThreadLocal允许一个线程以及该线程创建的所有子线程都可以访问它保存的值

代码示例

ThreadLocal使用

public class ThreadLocalTest {

    //第一种初始化方式
/**
* 声明为static是让ThreadLocal实例随着程序的结束才结束,这样才不会让GC回收了
* 声明为final是让ThreadLocal实例引用不会被替换,这样子也不会因为被替换导致被GC回收
* 这两个声明都是为了避免作为key的ThreadLocal对象没有外部强引用而导致被GC回收,从而导致内存泄漏的问题,因为ThreadLocalMap<ThreadLocal, Object>中的ThreadLocal
* 对象作为key是弱引用,会被GC回收。
*/
private static final ThreadLocal<String> threadLocalStr = ThreadLocal.withInitial(() -> "fresh"); private static AtomicInteger intGen = new AtomicInteger(0);
//第二种初始化方式
private static final ThreadLocal<Integer> threadLocalInt = new ThreadLocal<Integer>() {
@Override
public Integer initialValue() {
return intGen.incrementAndGet();
}
}; public static void main(String[] args) throws InterruptedException {
ArrayList<Thread> threads = new ArrayList<>();
for (int i = 0; i < 2; i++) {
Thread t = new Thread(() -> {
try {
System.out.println(Thread.currentThread().getName() + " " + threadLocalInt.get());
System.out.println(Thread.currentThread().getName() + " " + threadLocalStr.get());
TimeUnit.SECONDS.sleep(5);
threadLocalStr.set("bojack horseman" + threadLocalInt.get());
System.out.println(Thread.currentThread().getName() + " " + threadLocalInt.get());
System.out.println(Thread.currentThread().getName() + " " + threadLocalStr.get());
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
threadLocalInt.remove();
threadLocalStr.remove();
}
});
t.start();
threads.add(t);
}
TimeUnit.SECONDS.sleep(2);
System.out.println(threads);
System.out.println(threadLocalStr);
System.out.println(threadLocalInt);
}
/**
* Thread-0 1
* Thread-1 2
* Thread-0 fresh
* Thread-1 fresh
* [Thread[Thread-0,5,main], Thread[Thread-1,5,main]]
* java.lang.ThreadLocal$SuppliedThreadLocal@1ef7fe8e
* cn.vv.schedule.test.ThreadLocalTest$1@6f79caec
* Thread-1 2
* Thread-1 bojack horseman2
* Thread-0 1
* Thread-0 bojack horseman1
*/ }

InheritableThreadLocal使用

public class InheritableThreadLocalTest {

    //第一种初始化方式
private static final InheritableThreadLocal<String> threadLocalStr = new InheritableThreadLocal<String>() {
@Override
public String initialValue() {
return "fresh";
}
};
private static AtomicInteger intGen = new AtomicInteger(0);
//第二种初始化方式
private static final ThreadLocal<Integer> threadLocalInt = new ThreadLocal<Integer>() {
@Override
public Integer initialValue() {
return intGen.incrementAndGet();
}
}; public static void main(String[] args) throws InterruptedException {
//如果是InheritableThreadLocal,则父线程创建的所有子线程都会复制一份父线程的线程变量,而不是去初始化一份线程变量
threadLocalStr.set("main");
ArrayList<Thread> threads = new ArrayList<>();
for (int i = 0; i < 2; i++) {
Thread t = new Thread(() -> {
try {
System.out.println(Thread.currentThread().getName() + " " + threadLocalInt.get());
System.out.println(Thread.currentThread().getName() + " " + threadLocalStr.get());
TimeUnit.SECONDS.sleep(5);
//子线程可以自由地改变自己的本地变量
threadLocalStr.set("bojack horseman" + threadLocalInt.get());
System.out.println(Thread.currentThread().getName() + " " + threadLocalInt.get());
System.out.println(Thread.currentThread().getName() + " " + threadLocalStr.get());
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
threadLocalInt.remove();
threadLocalStr.remove();
}
});
t.start();
threads.add(t);
}
TimeUnit.SECONDS.sleep(2);
System.out.println(Thread.currentThread().getName() + " " + threadLocalStr.get());
}
/**
* Thread-0 2
* Thread-1 1
* Thread-0 main
* Thread-1 main
* main main
* Thread-0 2
* Thread-0 bojack horseman2
* Thread-1 1
* Thread-1 bojack horseman1
*/ }

参考

ThreadLocal理解及应用

ThreadLocal的正确使用与原理的更多相关文章

  1. 深入解析ThreadLocal 详解、实现原理、使用场景方法以及内存泄漏防范 多线程中篇(十七)

    简介 从名称看,ThreadLocal 也就是thread和local的组合,也就是一个thread有一个local的变量副本 ThreadLocal提供了线程的本地副本,也就是说每个线程将会拥有一个 ...

  2. 【java】ThreadLocal线程变量的实现原理和使用场景

    一.ThreadLocal线程变量的实现原理 1.ThreadLocal核心方法有这个几个 get().set(value).remove() 2.实现原理 ThreadLocal在每个线程都会创建一 ...

  3. ThreadLocal用法详解和原理(转)

    本文转自https://www.cnblogs.com/coshaho/p/5127135.html 感谢作者 一.用法 ThreadLocal用于保存某个线程共享变量:对于同一个static Thr ...

  4. volatile、ThreadLocal的使用场景和原理

    并发编程中的三个概念 原子性 一个或多个操作.要么全部执行完成并且执行过程不会被打断,要么不执行.最常见的例子:i++/i--操作.不是原子性操作,如果不做好同步性就容易造成线程安全问题. 可见性 多 ...

  5. 从源码看Thread&ThreadLocal&ThreadLocalMap的关系与原理

    1.三者的之间的关系 ThreadLocalMap是Thread类的成员变量threadLocals,一个线程拥有一个ThreadLocalMap,一个ThreadLocalMap可以有多个Threa ...

  6. ThreadLocal类详解:原理、源码、用法

    以下是本文目录: 1.从数据库连接探究 ThreadLocal 2.剖析 ThreadLocal 源码 3. ThreadLocal 应用场景 4. 通过面试题理解 ThreadLocal 1.从数据 ...

  7. ThreadLocal用法详解和原理

    一.用法 ThreadLocal用于保存某个线程共享变量:对于同一个static ThreadLocal,不同线程只能从中get,set,remove自己的变量,而不会影响其他线程的变量. 1.Thr ...

  8. Java并发编程:ThreadLocal的使用以及实现原理解析

    前言 前面的文章里,我们学习了有关锁的使用,锁的机制是保证同一时刻只能有一个线程访问临界区的资源,也就是通过控制资源的手段来保证线程安全,这固然是一种有效的手段,但程序的运行效率也因此大大降低.那么, ...

  9. windows系统下npm升级的正确姿势以及原理

    本文来自网易云社区 作者:陈观喜 网上关于npm升级很多方法多种多样,但是在windows系统下不是每种方法都会正确升级.其中在windows系统下主要的升级方法有以下三种: 首先最暴力的方法删掉no ...

随机推荐

  1. 洛谷 P6060 - [加油武汉]传染病研究(数论)

    洛谷题面传送门 一道不算太难的题,题解稍微写写吧( 首先根据约数个数和公式,对于一个 \(n=p_1^{\alpha_1}·p_2^{\alpha_2}·\cdots·p_m^{\alpha_m}\) ...

  2. 搭建FastDFS集群

    先插一张图(学习的时候找的)http://blog.csdn.net/u012453843/article/details/68957209?> 软件下载地址:主要是fastdfs.libfas ...

  3. Swift-技巧(十) Protocol 的灵活使用

    摘要 Protocol 是 Swift 中实现面向协议编程思想的重要部分.在使用过程中有遇到协议中声明的部分,但是在遵守部分不需要实现的,那么就需要使用 extension 参与进来,让 Protoc ...

  4. 脱离Editor、VS等IDE如何编译UE4工程

    在Windows平台下,我们从.uproject文件生成VS解决方案.sln文件 .uproject文件用于打开Editor .sln文件用于打开VS工程 对于有增加C++代码的工程,Editor中和 ...

  5. 用友低代码开发平台YonBuilder首次亮相DevRun开发者沙龙

    2020年的今天,没有人会再质疑企业上云的必要性与价值所在.从高科技行业到传统领域,大大小小的企业都希望走在变革道路前列,通过企业云加快业务数字化转型,更好地维护和管理企业数据. 然而,大多数企业都很 ...

  6. hive向mysql导入数据sqoop命令出错

    报错信息: java.lang.Exception: java.io.IOException: java.lang.ClassNotFoundException: info at org.apache ...

  7. Java项目发现==顺手改成equals之后,会发生什么?

    最近发生一件很尴尬的事情,在维护一个 Java 项目的时候,发现有使用 == 来比较两个对象的属性, 于是顺手就把 == 改成了 equals.悲剧发生...... == 和 equals 的区别 = ...

  8. 面对大规模 K8s 集群,这款诊断利器必须要“粉一波”!

    作者|段超 来源|尔达 Erda 公众号 背景 我们是一家做商业软件的公司,从一开始我们就把软件交付流程做的非常标准且简单,所有的软件都是基于我们的企业数字化平台 Erda(现已开源)来交付,底层基于 ...

  9. EDA简介

    Electronic design automation (EDA), also referred to as electronic computer-aided design (ECAD),[1] ...

  10. REMI源安装php7.3

    参考:https://blog.csdn.net/Phplayers/article/details/100901352 php5.6安装参考:https://www.cnblogs.com/Easo ...