线程学习记录07-ThreadLocal源码解析及实例

ThreadLocal :每个线程独立的局部变量

jdk1.8主要有三个方法:

set(); 为变量赋值
get();获取线程对应的变量
remove();移除该线程的变量

部分源码:
get()方法:

   public T get() {
    //获取当前线程(这作为key值)
    Thread t = Thread.currentThread();
    //获取这个线程对应的MAP
    ThreadLocalMap map = getMap(t);
     //如果该线程的map存在,则去获取该map的value值(当前ThreadLocal对象就是这个map的key)
    if (map != null) {
        ThreadLocalMap.Entry e = map.getEntry(this);
        if (e != null) {
            @SuppressWarnings("unchecked")
            T result = (T)e.value;
            return result;
        }
    }
    //如果不存在map,则进行初始化
    return setInitialValue();
}

 /**
  * 初始化ThreadLocalMap 的方法
  */
private T setInitialValue() {
    //这个方法固定返回的就是null
    T value = initialValue();
    //如果ThreadLocalMap 不存在,则就先初始化一个null的ThreadLocalMap 
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    if (map != null)
        map.set(this, value);
    else
        createMap(t, value);
    return value;
}

set()方法:

public void set(T value) {
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    if (map != null)
        //重点:this是map的key值
        map.set(this, value);
    else
        createMap(t, value);
}

void createMap(Thread t, T firstValue) {
    //产生一个map容器,this固定的作为key值。创建map之后将其赋予给线程,threadLocals 是Thread里面的一个变量
    t.threadLocals = new ThreadLocalMap(this, firstValue);
}

remove()方法:

public void remove() {
    //很简单,直接利用内部存在且已知的两个key删除对应的数据
     ThreadLocalMap m = getMap(Thread.currentThread());
     if (m != null)
         m.remove(this);
 }

JDK1.8的ThreadLocal 主要的就是以上三个方法。通过上面三个方法,我们可以知道的就是ThreadLocal为每个线程创建了一个ThreadLocalMap ,而ThreadLocal对象本身又去作为ThreadLocalMap 的key值,这样可以保证每个线程都能用有自己独立的变量。
ThreadLocal 与synchronized等加锁的方式不同,ThreadLocal 完全不提供锁,而使用空间换时间的手段,为每个线程提供独立的副本,保证线程的安全性。如果在并发不是很高的时候加锁的性能更好,但是并发量很高的时候使用ThreadLocal 可以一定程度的减少锁竞争。

实例代码:

public class ConnThreadLocal {

public static ThreadLocal<String> th = new ThreadLocal<String>();

public void setTh(String value){
    th.set(value);
}
public void getTh(){
    System.out.println(Thread.currentThread().getName() + ":" + this.th.get());
}

public static void main(String[] args) throws InterruptedException {

    final ConnThreadLocal ct = new ConnThreadLocal();
    Thread t1 = new Thread(new Runnable() {
        @Override
        public void run() {
            ct.setTh("张三");
            ct.getTh();
        }
    }, "t1");

    Thread t2 = new Thread(new Runnable() {
        @Override
        public void run() {
            try {
                Thread.sleep(1000);
                ct.getTh();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }, "t2");

    t1.start();
    t2.start();
}
}

运行结果:

image.png

t1线程已经为ThreadLocal赋值,可以获取到正确的值。t2获取的依旧是null,因为t2在ThreadLocal里面的Map并没被赋值

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 前言 ThreadLocal很多同学都搞不懂是什么东西,可以用来干嘛。但面试时却又经常问到,所以这次我和大家一起学...
    liangzzz阅读 12,511评论 14 228
  • Android Handler机制系列文章整体内容如下: Android Handler机制1之ThreadAnd...
    隔壁老李头阅读 7,684评论 4 30
  • Java SE 基础: 封装、继承、多态 封装: 概念:就是把对象的属性和操作(或服务)结合为一个独立的整体,并尽...
    Jayden_Cao阅读 2,150评论 0 8
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,376评论 11 349
  • 原创文章&经验总结&从校招到A厂一路阳光一路沧桑 详情请戳www.codercc.com 1. ThreadLoc...
    你听___阅读 6,768评论 8 19