初探 ThreadLocal
在多线程下最容易出现的问题就是并发导致共享变量被修改时出现的数据同步问题,为了解决该问题,一般的处理方式就是添加一个锁,通过锁机制,既可使得在多线程的环境之下解决共享资源部同步的问题,如图所示。
由于锁的引入,也使得多线程编程有了一定的门槛,加重了程序员在设计程序的负担。那么有没有一种方式,当每个线程创建一个变量后,访问的变量是属于当前线程自己的私有变量呢? ThreadLocal
的存在就可以完成该件事,ThreadLocal 的出现其实并不是为了解决以上的问题, ThreadLoca 是 JDK 包自己所提供的。
代码来自于《Java编程之美》1.11.1 ThreadLocal 使用示例小节
package com.example.thread;
/**
* 代码实例来自与《Java并发编程之美》1.11.1 ThreadLocal使用实例
*/
public class ThreadLocalTest {
//print打印函数
static void print(){
//当前线程本地内存中localVariable变量中的值
System.out.println("str"+localVariable.get());
//清除当前localVariabel变量中的值
//localVariable.remove();
}
static ThreadLocal<String> localVariable=new ThreadLocal<>();
Thread threadOne=new Thread(new Runnable() {
@Override
public void run() {
//设置线程中变量的值
localVariable.set("threadOne 线程的设置值");
System.out.println("获得线程threadOne中变量值:"+localVariable.get());
System.out.println("移除线程threadOne中变量的值:");
localVariable.remove();
System.out.println("再次获得线程threadOne中的变量的值:"+localVariable.get());
}
});
Thread threadTwo=new Thread(new Runnable() {
@Override
public void run() {
//设置线程中变量的值
localVariable.set("threadTwo 线程的设置值");
System.out.println("获得线程threadTwo中变量值:"+localVariable.get());
System.out.println("移除线程threadTwo中变量的值:");
localVariable.remove();
System.out.println("再次获得线程threadTwo中的变量的值:"+localVariable.get());
}
});
public static void main(String[] args) {
ThreadLocalTest localTest = new ThreadLocalTest();
localTest.threadOne.start();
localTest.threadTwo.start();
}
}
- 运行结果:
可以看到的是在运行的结果中 thraedOne
线程与 threadTwo
线程操作的是自己所属的私有变量 locaVariable
,那么关于 ThreadLocal
线程的底层到底是怎么实现的呢?为什么我们通过 set
与 get
方法就可以实现变量 localVariable
私有呢?
在 set
方法中 有一个 getMap 方法
public void set(T value) {
//获得当前线程对象
Thread t = Thread.currentThread();
//获得当前线程中的 thredadLocal 局部变量
ThreadLocalMap map = getMap(t);
if (map != null) {
map.set(this, value);
} else {
createMap(t, value);
}
}
在这里可以看到 getMap 是调用了 Thraed 类中的 thradLocals
方法获取当前的线程局部变量而 threadLocals
变量的底层是以为 Map
集合的方式进行线程局部变量的存储.
ThreadLocalMap getMap(Thread t) {
return t.threadLocals;
}
这里可以看到的 ThreadLocal
中 Thread 类中的 threadLocals
变量本身是 null
,threadLocals
变量是在调用 set
方法的时候才给赋值所得
/* ThreadLocal values pertaining to this thread. This map is maintained
* by the ThreadLocal class. */
ThreadLocal.ThreadLocalMap threadLocals = null;