Java ThreadLocal 示例

Java的ThreadLocal用于创建线程局部变量。我们知道,一个对象的所有线程共享其变量,因此该变量不是线程安全的。我们可以使用同步来确保线程安全,但如果我们想避免同步,我们可以使用ThreadLocal变量。

Java的ThreadLocal

每个线程都有自己的ThreadLocal变量,它们可以使用它的get()和set()方法来获取默认值或更改其在线程中的本地值。ThreadLocal实例通常是希望将状态与线程关联起来的类中的私有静态字段。

Java的ThreadLocal示例

以下是一个小例子,展示了在java程序中使用ThreadLocal并证明每个线程都有自己的ThreadLocal变量的副本。ThreadLocalExample.java

package com.journaldev.threads;

import java.text.SimpleDateFormat;
import java.util.Random;

public class ThreadLocalExample implements Runnable{

    // SimpleDateFormat不是线程安全的,因此为每个线程提供一个
    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.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的默认formatter与初始化的值相同。其他线程也显示相同的模式。更新: ThreadLocal类在Java 8中扩展了一个新方法withInitial(),该方法将Supplier功能接口作为参数。因此,我们可以使用lambda表达式轻松创建ThreadLocal实例。例如,上述的formatter ThreadLocal变量可以在一行中定义如下:

private static final ThreadLocal<SimpleDateFormat> formatter = 
	ThreadLocal.<SimpleDateFormat>withInitial
	(() -> {return new SimpleDateFormat("yyyyMMdd HHmm");});

如果您对Java 8的新功能不熟悉,请查看Java 8功能Java 8功能接口。这就是关于Java编程中的ThreadLocal的全部内容。参考:API文档

Source:
https://www.digitalocean.com/community/tutorials/java-threadlocal-example