Java ThreadLocal示例
时间:2020-02-23 14:36:58 来源:igfitidea点击:
Java ThreadLocal用于创建线程局部变量。
我们知道,对象的所有线程都共享其变量,因此该变量不是线程安全的。
我们可以使用同步来确保线程安全,但是如果要避免同步,可以使用ThreadLocal
变量。
Java ThreadLocal
每个线程都有自己的" ThreadLocal"变量,他们可以使用它的get()和set()方法获取默认值或者将其本地值更改为Thread。
ThreadLocal实例通常是希望将状态与线程关联的类中的私有静态字段。
Java ThreadLocal示例
这是一个小示例,展示了在Java程序中使用ThreadLocal并证明每个线程都有其自己的ThreadLocal变量副本。
ThreadLocalExample.java
package com.theitroad.threads; 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<SimpleDateFormat> formatter = new ThreadLocal<SimpleDateFormat>(){ @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更改了格式化程序的值,但线程2默认格式化程序仍与初始化值相同。
您也可以在其他线程中看到相同的模式。
更新:ThreadLocal类在Java 8中使用新方法withInitial()
进行了扩展,该方法以Supplier功能接口作为参数。
因此,我们可以使用lambda表达式轻松创建ThreadLocal实例。
例如,可以在以下一行中定义上述格式化程序ThreadLocal变量:
private static final ThreadLocal<SimpleDateFormat> formatter = ThreadLocal.<SimpleDateFormat>withInitial (() -> {return new SimpleDateFormat("yyyyMMdd HHmm");});