ThreadLocal工作原理及用法案例
作者:卡斯特梅的雨伞 发布时间:2021-11-13 04:22:39
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 回收,这个时候就可能会产生内存泄露。
因此针对这种情况,我们有两种原则:
ThreadLocal申明为private static final。JDK建议ThreadLocal定义为private static,这样ThreadLocal的弱引用问题则不存在了。
private与final 尽可能不让他人修改变更引用。
static 表示为类属性,只有在程序结束才会被回收。
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理解及应用
来源:https://www.cnblogs.com/castamere/p/15727414.html


猜你喜欢
- 摘 要1. 生成多个防伪码,防伪码的长度和个数由用户指定。2. 防伪码由"0123456789ABCDEFGHJKLMNPQRST
- web.xml中servlet> <servlet-name>SpringMVC</servlet-name>
- Java代码 mvn install:install-file -DgroupId=包名 -DartifactId=项目名 -Dversio
- Java中常用关键字:与数据类型相关(10)与流程控制相关(13)if: 表示条件判断,一般用法if(关系表达式),后跟else或
- 目录有状态组件异步 async/await引入 flutter_easyrefresh使用 flutter_easyrefresh运行结果结
- 1.设置url-pattern为*.do(最为常见的方式)只要你的请求url中包含配置的url-pattern,该url就可以到达Dispa
- 1. 场景描述本节结合springboot2、springmvc、mybatis、swagger2等,搭建一个完整的增删改查项目,希望通过这
- 这篇文章主要介绍了java内存泄漏与内存溢出关系解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友
- 1. SpringBoot 配置文件1.1 配置文件的作用配置文件中配置了项目中重要的数据, 例如:数据库的连接信息 (用户名密码)项目的启
- 一、Map接口继承树Map:双列数据,存储key-value对的数据 ---类似于高中的函数:y = f(x)A.HashMap:作为Map
- 综述在Android系统中,出于对性能优化的考虑,对于Android的UI操作并不是线程安全的。也就是说若是有多个线程来操作UI组件,就会有
- 从一个Stream中过滤null值复习一个Stream 包含 null 数据的例子.Java8Examples.javapackage co
- 1. 要求获取指定文件夹下(包含子文件夹),所有指定后缀(如txt)的文件路径(即文件所在目录+文件名),返回一个字符串数组。2. 代码获取
- 一、图示二、多线程编程何为多线程,通俗的讲就是让你的代码同时干好几件事。而我们的一个代码文件或者一个项目就是一个进程,而如果我们想提高效率,
- 本文实例讲述了Java线程池用法。分享给大家供大家参考,具体如下:一 使用newSingleThreadExecutor创建一个只包含一个线
- 使用NOPI导入Excel文档NOPI版本:2.3.0,依赖于NPOI的SharpZipLib版本:0.86,经测试适用于.net4.0+记
- 概述: 当希望能直接在数据库语言中只检索符合条件的记录,不需要再通过程序对其做处理时,SQL语句分页
- 一、获取接口请求的数据可以在Interceptor的afterCompletion中实现但是要重写RequestWrapper代码记录如下:
- java中多种方式读文件 一、多种方式读文件内容。 1、按字节读取文件内容 2、按字符读取文件内容 3、按行读取文件内容 4、随机读取文件内
- Java中的Static class详解Java中的类可以是static吗?答案是可以。在Java中我们可以有静态实例变量、静态