这篇文章主要介绍了ThreadLocal怎么用,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。
1. ThreadLocal<T> 简介和使用示例
ThreadLocal只有一个无参的构造方法
public ThreadLocal()
ThreadLocal的相关方法
public T get()
public void set(T value)
public void remove()
protected T initialValue()
initialValue方法的访问修饰符是protected,该方法为第一次调用get方法提供一个初始值。默认情况下,第一次调用get方法返回值null。在使用时,我们一般会复写ThreadLocal的initialValue方法,使第一次调用get方法时返回一个我们设定的初始值。
下面是一个ThreadLocal的一个简单使用示例
package javalearning;import java.util.Random;import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors;import java.util.concurrent.Semaphore;public class ThreadLocalDemo {private ThreadLocal<Integer> tlA = new ThreadLocal<Integer>(){protected Integer initialValue(){return 3;}};Semaphore semaphore = new Semaphore(1);private Random rnd = new Random();public class Worker implements Runnable{@Override public void run(){try {Thread.sleep(rnd.nextint(1000));semaphore.acquire();}catch (InterruptedException e) {e.printStackTrace();}int valA = tlA.get();System.out.println(Thread.currentThread().getName() +" tlA initial val : "+ valA);valA = rnd.nextint();tlA.set(valA);System.out.println(Thread.currentThread().getName() +" tlA new val: "+ valA);semaphore.release();tlA.remove();}}public static void main(String[] args){ExecutorService es = Executors.newFixedThreadPool(3);ThreadLocalDemo tld = new ThreadLocalDemo();es.execute(tld.new Worker());es.execute(tld.new Worker());es.execute(tld.new Worker());es.shutdown();}}
运行结果
pool-1-thread-1 tlA initial val : 3pool-1-thread-1 tlA new val: -1288455998pool-1-thread-3 tlA initial val : 3pool-1-thread-3 tlA new val: 112537197pool-1-thread-2 tlA initial val : 3pool-1-thread-2 tlA new val: -12271334
从运行结果可以看出,每个线程第一次调用TheadLocal对象的get方法时都得到初始值3,注意我们上面的代码是让三个线程顺序执行,显然从运行结果看,pool-1-thread-1线程结束后设置的新值,对pool-1-thread-3线程是没有影响的,pool-1-thread-3线程完成后设置的新值对pool-1-thread-2线程也没有影响。这就仿佛把ThreadLocal对象当做每个线程内部的对象一样,但实际上tlA对象是个外部类对象,内部类Worker访问到的是同一个tlA对象,也就是说是被各个线程共享的。这是如何做到的呢?我们现在就来看看ThreadLocal对象的内部原理。
2.ThreadLocal<T>的原理
首先,在Thread类中定义了一个threadLocals,它是ThreadLocal.ThreadLocalMap对象的引用,默认值是null。ThreadLocal.ThreadLocalMap对象表示了一个以开放地址形式的散列表。当我们在线程的run方法中第一次调用ThreadLocal对象的get方法时,会为当前线程创建一个ThreadLocalMap对象。也就是每个线程都各自有一张独立的散列表,以ThreadLocal对象作为散列表的key,set方法中的值作为value(第一次调用get方法时,以initialValue方法的返回值作为value)。显然我们可以定义多个ThreadLocal对象,而我们一般将ThreadLocal对象定义为static类型或者外部类中。上面所表达的意思就是,相同的key在不同的散列表中的值必然是独立的,每个线程都是在各自的散列表中执行操作。
TheadLocal中的get源代码
public T get() { Thread t = Thread.currentThread(); ThreadLocalMap map = getMap(t); if (map != null) { ThreadLocalMap.Entry e = map.getEntry(this);//这里的this是指当前的ThreadLocal对象 if (e != null) { @SuppressWarnings("unchecked") T result = (T)e.value; return result; } } return setInitialValue();}
感谢你能够认真阅读完这篇文章,希望小编分享的“ThreadLocal怎么用”这篇文章对大家有帮助,同时也希望大家多多支持编程网,关注编程网行业资讯频道,更多相关知识等着你来学习!