ThreadLocal详解
1. ThreadLocal
- 定义:提供线程局部变量;一个线程局部变量在多个线程中,分别有独立的值(副本)。
- 特点:简单(开箱即用)、快速(无额外开销)、安全(线程安全)。
- 场景:多线程场景(资源持有、线程一致性、并发计算、线程安全等场景)。
- 实现原理:java中使用哈希表实现。
- 应用范围:几乎所有提供多线程特征的语言。
2. ThreadLocal API
- get方法:
public class ThreadLocalAPi1 {
private static ThreadLocal<Long> threadLocal = new ThreadLocal<Long>() {
/**
* Note1:
* 只有ThreadLocal.get()方法时, 且只有当前ThreadLocal中没有值的时候,
* 才会触发initialValue().
*/
@Override
protected Long initialValue() {
System.out.println("initial value...");
return 100L;
}
};
public static void main(String[] args) {
System.out.println(threadLocal.get());
}
}
- 其它API
remove:移除当前线程中的数据。
set:给当前线程设置值。
3. ThreadLocal 使用场景
- 线程资源持有:供线程的各个部分使用,全局获取,减少编程难度。
- 线程资源一致性:帮助需要保持线程一致的资源(如数据库事务)维护一致性,降低编程难度。
- 线程安全:帮助只考虑了单线程的程序库,无缝向多线程场景迁移。
- 分布式计算:帮助分布式计算场景的各个线程累计局部计算结果。
4. 涉及ThreadLocal的框架
-
Quartz:
- SimpleSemaphore提供资源隔离。
- SimpleSemaphore中的lockOwners(ThreadLocal)为重度锁操作前置过滤。
-
Mybaits:
- SqlSessionMaagger#localSqlSession:在一个事务中,要保证事务,就必须所有的jdbc操作是在同一个连接中。所有的线程的连接都存放在这个ThreadLocal中。
- SqlSessionMaagger#localSqlSession:在一个事务中,要保证事务,就必须所有的jdbc操作是在同一个连接中。所有的线程的连接都存放在这个ThreadLocal中。
-
Spring分布式事务:TransactionContextHolder#currentTransactionContext就是使用ThreadLocal。
5. ThreadLocal源码分析
- 使用的hash算法,hash因子为’ox61c88647’