1.ThreadLocal介绍
ThreadLocal是Java中一个非常有用的工具类,它可以让每个线程维护自己的数据副本,确保线程安全性。
在多线程编程中,如果多个线程共享同一个变量,就需要使用同步控制来保证线程安全,这会带来很大的性能开销。而ThreadLocal可以让每个线程都拥有自己的变量副本,这样就不需要进行同步控制,从而提高了程序的并发性能。
2.ThreadLocal常见应用场景
Web应用中,每个HTTP请求都会被一个线程处理,可以使用ThreadLocal来存储当前请求的用户信息或者权限信息,以避免在每个方法中都进行传参。
在线程池中执行任务时,每个线程都会被多个任务复用,可以使用ThreadLocal来避免在每个任务中都创建对象或者初始化资源。
在某些场景下,需要在一个线程中传递某个对象或者数据到另一个线程中,可以使用ThreadLocal来存储这个对象或者数据,从而在另一个线程中获取。
但在实际开发中,Web应用可用coockie在浏览器上存储信息,线程池已经有现成的工具了,根本用不到ThreadLocal,因此我举一个ThreadLocal在实际开发中的例子
3.ThreadLocal在金融领域应用
//金融股票类计算器
public class StockCalculator {
//股票价格
private final BigDecimal stock;
//费率k
private final BigDecimal k;
//本金
private final BigDecimal price;
public StockCalculator() {
this.stock = BigDecimal.valueOf(RandomUtil.randomInt(10, 20));
this.k = BigDecimal.valueOf(RandomUtil.randomDouble(0, 0.5));
this.price = BigDecimal.valueOf(5);
}
// 计算 A,假设每次计算需要3分钟
public BigDecimal doA(){
BigDecimal multiply = stock.multiply(k);
System.out.println(Thread.currentThread() + "A:"+ multiply);
return multiply;
}
// 计算 B,假设每次计算需要5分钟
public BigDecimal doB(){
BigDecimal subtract = stock.subtract(price);
System.out.println(Thread.currentThread() + "B:"+ subtract);
return subtract;
}
}
假设这里有一个【金融股票类计算器】,客户需要看到股票价格stock,费率k所计算出来的A,B,而stock,k在股票交易中每一秒都是不一样的,。
试想:如果计算完A之后,股票价格stock,费率k发生变化,再去计算B,则A,B就没有业务参考价值。
因此这里就需要用到ThreadLocal:
public class Main {
public static void main(String[] args) {
ThreadLocal<StockCalculator> stockCalculatorThreadLocal = ThreadLocal.withInitial(StockCalculator::new);
for (int i = 0; i < 10; i++) {
new Thread(()->{
StockCalculator stockCalculator = stockCalculatorThreadLocal.get();
stockCalculator.doA();
stockCalculator.doB();
}).start();
}
}
}
这里为每个线程创建一个计算器副本,计算器内部的值价格stock,费率k在每个副本可能都是不一样的,但又互不影响,这样计算出来的A,B才有业务价值。
4.ThreadLocal的优势
ThreadLocal可以让我们更方便地在多线程环境下维护线程的局部变量,避免了使用同步控制带来的性能损失,同时也减少了代码的复杂度。但是需要注意的是,使用ThreadLocal要注意内存泄漏的问题,需要在适当的时候清理ThreadLocal变量。