普通对象的变量,当有多个线程访问时,虽然每个线程都有一份拷贝,但这是由于虚拟机优化而发生的,原则上来看都是同一份。可以通过加volatitle关键字阻止虚拟机优化。本文讨论的相反,希望实现每一个线程都有自己专有的变量对象。您自己当然可以通过threadId+Map的方式实现,但Java类库已经提供了ThreadLocal,可以直接使用,本文将介绍其用法。
ThreadLocal是一个泛型类,可以设置任意类型的对象。它对主要的接口是set和get方法,写和读线程专属变量。普通的用法很简单,先上代码:
import java.util.Random;
public class ThreadLocalTest {
private static ThreadLocal<Integer> threadIntegerVariable = new ThreadLocal<Integer>();
public static void main(String[] args) {
for (int i = 0; i < 20; i++) {
CustomerThread customerThread = new CustomerThread();
customerThread.start();
}
}
static class CustomerThread extends Thread {
@Override
public void run() {
Random random = new Random();
Integer threadSequence = random.nextInt();
threadIntegerVariable.set(threadSequence);
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
printThreadVariable();
}
}
private static void printThreadVariable() {
Long threadId = Thread.currentThread().getId();
Integer threadSequence = threadIntegerVariable.get();
System.out.println(threadId + " : " + threadSequence);
}
}
首先声明一个具体内容类型为Integer的ThreadLocal变量threadIntegerVariable,threadIntegerVariable自身所有线程都能够访问得到(共用),但是当不同的线程调用threadIntegerVariable的set方法时,设置的内容是线程独有的,不同的线程有不同的内容对象,在此例中就是有不同的Integer对象,当然就是不同的值。如果您希望存储其它类型的对象,把Integer换为您希望的类型即可。
该用法在需要实现线程专有变量时非常方便和有效。线程日志,线程事务或其它专有变量都可以使用。