Java ThreadLocal用于创建线程局部变量。我们知道对象的所有线程都共享它的变量,因此变量不是线程安全的。可以使用同步来实现线程安全,但如果想避免同步,可以使用ThreadLocal变量。
每个线程都有自己的ThreadLocal变量,可以使用它的get()和set()方法来获取默认值或将其值更改为Thread的本地值。
ThreadLocal实例通常是希望将状态与线程相关联的类中的私有静态字段。
Java ThreadLocal示例
下面是一个小例子,演示如何在java程序中使用ThreadLocal并证明每个线程都有自己的ThreadLocal变量副本。
import java.text.SimpleDateFormat;
import java.util.Random;
public class ThreadLocalExample implements Runnable{
// SimpleDateFormat is not thread-safe, so give one to each thread
private static final ThreadLocal formatter = new ThreadLocal(){
@Override
protected SimpleDateFormat initialValue()
{
return new SimpleDateFormat("yyyyMMdd HHmm");
}
};
public static void main(String[] args) throws InterruptedException {
ThreadLocalExample obj = new ThreadLocalExample();
for(int i=0 ; i<10; i++){
Thread t = new Thread(obj, ""+i);
Thread.sleep(new Random().nextInt(1000));
t.start();
}
}
@Override
public void run() {
System.out.println("Thread Name= "+Thread.currentThread().getName()+" default Formatter = "+formatter.get().toPattern());
try {
Thread.sleep(new Random().nextInt(1000));
} catch (InterruptedException e) {
e.printStackTrace();
}
//formatter pattern is changed here by thread, but it won't reflect to other threads
formatter.set(new SimpleDateFormat());
System.out.println("Thread Name= "+Thread.currentThread().getName()+" formatter = "+formatter.get().toPattern());
}
}
上面的java ThreadLocal示例程序的输出是:
Thread Name= 0 default Formatter = yyyyMMdd HHmm
Thread Name= 1 default Formatter = yyyyMMdd HHmm
Thread Name= 0 formatter = M/d/yy h:mm a
Thread Name= 2 default Formatter = yyyyMMdd HHmm
Thread Name= 1 formatter = M/d/yy h:mm a
Thread Name= 3 default Formatter = yyyyMMdd HHmm
Thread Name= 4 default Formatter = yyyyMMdd HHmm
Thread Name= 4 formatter = M/d/yy h:mm a
Thread Name= 5 default Formatter = yyyyMMdd HHmm
Thread Name= 2 formatter = M/d/yy h:mm a
Thread Name= 3 formatter = M/d/yy h:mm a
Thread Name= 6 default Formatter = yyyyMMdd HHmm
Thread Name= 5 formatter = M/d/yy h:mm a
Thread Name= 6 formatter = M/d/yy h:mm a
Thread Name= 7 default Formatter = yyyyMMdd HHmm
Thread Name= 8 default Formatter = yyyyMMdd HHmm
Thread Name= 8 formatter = M/d/yy h:mm a
Thread Name= 7 formatter = M/d/yy h:mm a
Thread Name= 9 default Formatter = yyyyMMdd HHmm
Thread Name= 9 formatter = M/d/yy h:mm a
从输出中可以看出,Thread-0已经改变了formatter的值,但thread-2仍然是默认格式化程序的初始化值。也可以看到其他线程的相同模式。
更新:ThreadLocal类在Java 8中使用新方法withInitial()进行扩展,该方法将Supplier功能接口作为参数。因此可以使用lambda表达式创建ThreadLocal实例。例如,上面的格式化程序ThreadLocal变量可以在一行中定义如下:
private static final ThreadLocal formatter =
ThreadLocal.withInitial
(() -> {return new SimpleDateFormat("yyyyMMdd HHmm");});
如果不熟悉Java 8的功能,请查看Java 8功能和Java 8功能接口。
¥ 我要打赏
纠错/补充
收藏
加QQ群啦,易百教程官方技术学习群
注意:建议每个人选自己的技术方向加群,同一个QQ最多限加 3 个群。