ThreadLocal的正确使用与原理
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的正确使用与原理的更多相关文章
- 深入解析ThreadLocal 详解、实现原理、使用场景方法以及内存泄漏防范 多线程中篇(十七)
简介 从名称看,ThreadLocal 也就是thread和local的组合,也就是一个thread有一个local的变量副本 ThreadLocal提供了线程的本地副本,也就是说每个线程将会拥有一个 ...
- 【java】ThreadLocal线程变量的实现原理和使用场景
一.ThreadLocal线程变量的实现原理 1.ThreadLocal核心方法有这个几个 get().set(value).remove() 2.实现原理 ThreadLocal在每个线程都会创建一 ...
- ThreadLocal用法详解和原理(转)
本文转自https://www.cnblogs.com/coshaho/p/5127135.html 感谢作者 一.用法 ThreadLocal用于保存某个线程共享变量:对于同一个static Thr ...
- volatile、ThreadLocal的使用场景和原理
并发编程中的三个概念 原子性 一个或多个操作.要么全部执行完成并且执行过程不会被打断,要么不执行.最常见的例子:i++/i--操作.不是原子性操作,如果不做好同步性就容易造成线程安全问题. 可见性 多 ...
- 从源码看Thread&ThreadLocal&ThreadLocalMap的关系与原理
1.三者的之间的关系 ThreadLocalMap是Thread类的成员变量threadLocals,一个线程拥有一个ThreadLocalMap,一个ThreadLocalMap可以有多个Threa ...
- ThreadLocal类详解:原理、源码、用法
以下是本文目录: 1.从数据库连接探究 ThreadLocal 2.剖析 ThreadLocal 源码 3. ThreadLocal 应用场景 4. 通过面试题理解 ThreadLocal 1.从数据 ...
- ThreadLocal用法详解和原理
一.用法 ThreadLocal用于保存某个线程共享变量:对于同一个static ThreadLocal,不同线程只能从中get,set,remove自己的变量,而不会影响其他线程的变量. 1.Thr ...
- Java并发编程:ThreadLocal的使用以及实现原理解析
前言 前面的文章里,我们学习了有关锁的使用,锁的机制是保证同一时刻只能有一个线程访问临界区的资源,也就是通过控制资源的手段来保证线程安全,这固然是一种有效的手段,但程序的运行效率也因此大大降低.那么, ...
- windows系统下npm升级的正确姿势以及原理
本文来自网易云社区 作者:陈观喜 网上关于npm升级很多方法多种多样,但是在windows系统下不是每种方法都会正确升级.其中在windows系统下主要的升级方法有以下三种: 首先最暴力的方法删掉no ...
随机推荐
- 联盛德 HLK-W806 (四): 软件SPI和硬件SPI驱动ST7735液晶LCD
目录 联盛德 HLK-W806 (一): Ubuntu20.04下的开发环境配置, 编译和烧录说明 联盛德 HLK-W806 (二): Win10下的开发环境配置, 编译和烧录说明 联盛德 HLK-W ...
- 『MdOI R1』Treequery
我们可以思考怎么做呢. 首先我们需要进行一些分类讨论: 我们先思考一下如果所有关键点都在 \(p\) 的子树内, 那显然是所有关键点的 \(Lca\) 到 \(p\) 距离. 如果所有关键点一些在 \ ...
- 富集分析DAVID、Metascape、Enrichr、ClueGO
前言 一般我们挑出一堆感兴趣的基因想临时看看它们的功能,需要做个富集分析.虽然公司买了最新版的数据库,如KEGG,但在集群跑下来嫌麻烦.这时网页在线或者本地化工具派上用场了. DAVID DAVID地 ...
- RNA-seq 生物学重复相关性验证
根据拿到的表达矩阵设为exprSet 1.用scale 进行标准化 数据中心化:数据集中的各个数字减去数据集的均值 数据标准化:中心化之后的数据在除以数据集的标准差. 在R中利用scale方法来对数据 ...
- linux下定位异常消耗的线程实战分析
前言: 之前分享过一篇Linux开发coredump文件分析实战分享 ,今天再来分享一篇实战文章. 在我们嵌入式linux开发过程中,开发过程中我们经常会使用多进程.多线程开发.那么多线程使用过程中, ...
- ASP.NET Core中使用固定窗口限流
算法原理 固定窗口算法又称计数器算法,是一种简单的限流算法.在单位时间内设定一个阈值和一个计数值,每收到一个请求则计数值加一,如果计数值超过阈值则触发限流,如果达不到则请求正常处理,进入下一个单位时间 ...
- Scala(七)【异常处理】
目录 一.try-catch-finally 二.Try(表达式).getOrElse(异常出现返回的默认值) 三. 直接抛出异常 一.try-catch-finally 使用场景:在获取外部链接的时 ...
- myatoi
atoi (表示 ascii to integer)是把字符串转换成整型数的一个函数,应用在计算机程序和办公软件中.int atoi(const char *nptr) 函数会扫描参数 nptr字符串 ...
- JVM——对象已“死”的判定
主要针对Java堆和方法区 1.判断对象是否已"死" Java堆中存放着几乎所有的对象实例,垃圾回收器在对堆进行回收之前,首先应该判断这些对象哪些还"存活",哪 ...
- 视频框架 Vitamio使用
转自http://blog.csdn.net/u010181592/article/category/5893483 1.在https://github.com/yixia/VitamioBundle ...