什么是ThreadLocal
ThreadLocal是java中的线程本地变量,用于每个线程保存自己的变量,使线程间的变量互不干扰。
ThreadLocal实现
每个线程内部都会维护一个ThreadLocalMap对象,该map的key为存储的变量对象,value为对象的值。
当调用ThreadLocal的set()方法时,如果map非空,则将ThreadLocal对象和值存入map中;如果map为空,则新建ThreadLocalMap
调用get()方法时,则先获取当前线程的ThreadLocal对象,再根据对象获取对应的value值。
*具体实现逻辑更为复杂,详细实现可以参考https://www.jianshu.com/p/98b68c97df9b
ThreadLocal使用案例
/** * 通过ThreadLocal保存各自线程中的变量 * Created by xuec on 2019/3/4. */ public class Test { //初始化变量 private ThreadLocal<Long> longLocal = new ThreadLocal<>(); private ThreadLocal<String> stringLocal = new ThreadLocal<>(); /** * 为变量赋值 */ private void set() { longLocal.set(Thread.currentThread().getId()); stringLocal.set(Thread.currentThread().getName()); } /** * 获取变量 * @return Long型变量 */ private Long getLong() { return longLocal.get(); } /** * 获取变量 * @return String型变量 */ private String getString() { return stringLocal.get(); } public static void main(String[] args) throws InterruptedException { final Test test = new Test(); //在主线程中赋值 test.set(); System.out.println("main-id: " + test.getLong()); System.out.println("main-name: " + test.getString()); //新建线程 final Thread thread1 = new Thread() { @Override public void run() { //在线程1中赋值 test.set(); //打印线程1中的值 System.out.println("Thread1-id: " + test.getLong()); System.out.println("Thread1-name: " + test.getString()); } }; thread1.start(); thread1.join(); //再次打印变量 System.out.println("main-id: " + test.getLong()); System.out.println("main-name: " + test.getString()); } }
输出:
1
2
3
4
5
6
|
main-id: 1 main-name: main Thread1-id: 11 Thread1-name: Thread- 0 main-id: 1 main-name: main |
从结果可以看出,main线程和Thread1线程中,longLocal和stringLocal保存的值都不一样。
ThreadLocal应用场景
ThreadLocal主要应用于数据库连接、线程会话等场景,每个线程都保持独立的会话。如果是变量是全局共享的,则并不适用于ThreadLocal。