Java线程ThreadLocal功能实例介绍



Java线程ThreadLocal功能实例介绍。ThreadLocal的功能是实现一个线程独立的全局变量。使用全局变量的好处是,程序的任何地方都可以使用它,而无需进行参数传递。在单线程环境下,将类的属性和方法定义为static,即可实现全局变量。但在多线程环境下,普通的全局变量是线程间共享的,一个线程修改了全局变量的值,另一个线程可以看到这个修改。使用ThreadLocal实现全局变量,即保留了随处可用这一优点,又能做到线程间独立,即线程A拥有线程A的全局变量,线程B拥有线程B的全局变量,互不影响。换句话说,只有当你需要在多线程环境下使用全局变量的时候,才会用到ThreadLocal,其他情况请使用别的同步工具。

实例:定义全局变量IdGenerator,负责得到线程的ID,不同线程使用该类,得到不同的ID,并且能够保持住该ID不发生变化。

public class ThreadLocalTest {

public static void main(String[] args) {
// 创建5个线程并依此启动,每个线程中调用业务逻辑处理相关信息
for(int i=0; i<5; i++) {
new Thread() {

@Override
public void run() {
new CommercialLogic().doTask();
}
}.start();
}
}

}


/**
* 此类中全是静态属性和静态方法
* 相当于全局变量
*/
class IdGenerator {
private final static AtomicInteger num = new AtomicInteger(1000);
private final static ThreadLocal<Integer> id = new ThreadLocal<Integer>() {
@Override
protected Integer initialValue() {
return num.incrementAndGet();
}
};

public static int getID() {
return id.get();
}
}
/**
* 业务逻辑类
* 在处理业务逻辑时,需要用到线程ID
*/
class CommercialLogic {
public void doTask() { //虽然要用到线程ID,但是在方法签名中并不需要传入该参数
for(int i=0; i<100; i++) {
//通过全局变量使用线程ID
System.out.println(Thread.currentThread().getName()+”线程的ID是”+IdGenerator.getID());
}
}
}

上例中,各个业务逻辑在使用线程ID的时候,无需通过参数传递进来,而是直接使用全局变量。同时,在业务逻辑中,打印显示线程ID100次,每个线程的ID都保持不变。若使用下面代码用普通方式实现的 IdGenerator作为全局变量,则每个线程的ID无法保持一致,线程间会相互影响。

// 普通的全局变量,错误,线程间会相互影响
class IdGenerator {
private static int i = 1000;

public static int getID() {
return ++i;
}
}