A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© 逆风TO 黑马粉丝团   /  2020-3-26 10:04  /  1075 人查看  /  0 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

ThreadLocal类的作用:为每个线程创建独立的副本,从而保证了线程安全。
ThreadLocal使用代码示例:
[JavaScript] 纯文本查看 复制代码
public class MyThreadLocalTest {
    private ThreadLocal<Integer> threadLocal=new ThreadLocal<Integer>(){
        @Override
        protected Integer initialValue() {
            return 1;
        }
    };

    public void StartThreadArray(){
        Thread[] runs = new Thread[5];
        for(int i=0;i<runs.length;i++){
            runs[i]=new Thread(new MyThreadLocalTest.TestThread(1));
        }
        for(int i=0;i<runs.length;i++){
            runs[i].start();
        }
    }

    public  class TestThread implements Runnable{
        int id;
        public TestThread(int id){
            this.id = id;
        }
        public void run() {
            System.out.println("this is :"+Thread.currentThread().getName()+":start");
            Integer s = threadLocal.get();
            s = s+id;
            threadLocal.set(s);
            System.out.println("this is :"+Thread.currentThread().getName()
                    +":"+ threadLocal.get());
        }
    }

    public static void main(String[] args){
        MyThreadLocalTest test = new MyThreadLocalTest();
        test.StartThreadArray();
    }
}

源码解析:
当使用ThreadLocal时,线程第一次调用ThreadLocal的get()方时,ThreadLocal对象会先获取当前线程对象,然后根据调用getMap(Thread t) 方法,从线程中获取到ThreadLocalMap对象(这个对象其实就是线程中的ThreadLocal对象和TheadLocal中的元素的绑定关系,类似于一个Map集合),第一次调用get()方法时该对象为空null,因为还未建立映射关系,此时对调用到setInitialValue()。
[JavaScript] 纯文本查看 复制代码
 public T get() {
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null) {
            ThreadLocalMap.Entry e = map.getEntry(this);
            if (e != null) {
                @SuppressWarnings("unchecked")
                T result = (T)e.value;
                return result;
            }
        }
        return setInitialValue();
    }


该方法会首先调用到initialValue(),你定义初始化的值,然后从线程中获取ThreadLocalMap对象,判断对象是否为空,如果为空则新建一个ThreadLocalMap对象并且调用createMap(Thread t,T value)方法。
该方法主要的作用是创建一个ThreadLocalMap对象,并且建立当前ThreadLocal对象以及初始化节点的对应关系,并且将这个ThreadLocalMap对象放入线程中的。
如果线程中的ThreadLocalMap对象不为空,则将当前ThreadLocal对象和初始化节点的映射关系放入ThreadLocalMap中。
[JavaScript] 纯文本查看 复制代码
private T setInitialValue() {
    T value = initialValue();
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    if (map != null)
        map.set(this, value);
    else
        createMap(t, value);
    return value;
}


这就是线程第一次使用get方法。当线程第二次调用get()方法时,就直接从线程中的ThreadLocalMap中获取值。
具体调用流程为:

3.png

0 个回复

您需要登录后才可以回帖 登录 | 加入黑马