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 ...
随机推荐
- 矩阵树定理&BEST定理学习笔记
终于学到这个了,本来准备省选前学来着的? 前置知识:矩阵行列式 矩阵树定理 矩阵树定理说的大概就是这样一件事:对于一张无向图 \(G\),我们记 \(D\) 为其度数矩阵,满足 \(D_{i,i}=\ ...
- Codeforces 1491H - Yuezheng Ling and Dynamic Tree(分块)
Codeforces 题目传送门 & 洛谷题目传送门 *3400 的毒瘤 H 题,特意写个题解纪念一下( 首先对于这种数据结构不太好直接维护的东东可以考虑分块.然鹅我除了分块其他啥也没想到 我 ...
- FESTUNG模型介绍—1.对流方程求解
FESTUNG模型介绍-1.对流方程求解 1. 控制方程 对流问题中,控制方程表达式为 \[\partial_t C + \partial_x (u^1 C) + \partial_y (u^2 C) ...
- 【机器学习与R语言】13- 如何提高模型的性能?
目录 1.调整模型参数来提高性能 1.1 创建简单的调整模型 2.2 定制调整参数 2.使用元学习来提高性能 2.1 集成学习(元学习)概述 2.2 bagging 2.3 boosting 2.4 ...
- Linux服务器I/O性能分析-2
一.如何正确分析IO性能 1.1 BLKTRACE分析IO性能 之前的文章已经说明,要是系统发生I/O性能问题,我们常用的命令是无法精确定位问题(内核I/O调度器消耗的时间和硬件消耗的时间,这个不能作 ...
- 26-Palindrome Number
回文数的判定,不使用额外的空间 Determine whether an integer is a palindrome. Do this without extra space. 思路:将一个整数逆 ...
- kubernetes整个基础环境的准备
1.三台centos7,用CentOS-7-x86_64-Minimal-1708.iso安装的,记得统一选好时区,这三台会有etcd集群,其中一台做kubernetes服务端(也可以做三台服务端做负 ...
- Selenium的安装和使用
一.Selenium的安装,Selenium是一个自动化测试工具,利用它我们可以驱动浏览器执行特定的动作,如点击.下拉等操作.对于一些JavaScript渲染的页面来说,这种抓取方式非常有效.1.pi ...
- typora 图床配置方法
学习计算机的同学,在日常学习中难免会记笔记,写文档.相信大家记笔记大部分使用的都是 Markdown 吧,如果到现在还没接触,那我强烈建议你去学习一下,大概几分钟就可以搞定它. 注:下文用到的所有软件 ...
- 关于Stream的使用
引言 Stream 是 Java8 中处理集合的关键抽象概念,它可以指定你希望对集合进行的操作,可以执行非常复杂的查找.过滤和映射数据等操作.使用Stream API 对集合数据进行操作,就类似于使用 ...