是什么
ThreadLocal提供线程局部变量。这些变量与正常的变量不同,因为每一个线程在访问ThreadLocal实例的时候(通过其get或set方法)都有自己的、独立初始化的变量副本。ThreadLocal实例通常是类中的私有静态字段,使用它的目的是希望将状态(例如,用户ID或事务ID)与线程关联起来
能干嘛
实现没一个线程都有自己专属的本地变量副本(自己用自己的变量不麻烦别人,不和其他人共享,人人有份,人各一份),主要解决了让每个线程绑定自己的值,通过get()和set()方法,获取默认值或将其更改为当前线程所存副本的值从而避免了线程安全问题,比如我们之前讲解的8锁案例,资源类是使用同一部手机,多线程抢夺同一部手机使用,假如人手一份是不是天下太平?
Hello world
不加锁解决线程安全
class SellHouse{
private static int threadNum;
// 推荐使用
/* ThreadLocal<Integer> threadLocal = new ThreadLocal<Integer>(){
@Override
protected Integer initialValue() {
return 0;
}
};*/
// java8
ThreadLocal<Integer> saleVolume = ThreadLocal.withInitial(() -> 0);
public void sale(){
saleVolume.set(1+saleVolume.get());
}
public static void main(String[] args) throws InterruptedException {
SellHouse sellHouse = new SellHouse();
threadNum = 5;
CountDownLatch downLatch = new CountDownLatch(threadNum);
for (int i = 0; i < threadNum; i++) {
new Thread(() -> {
try {
int size = new Random().nextInt(10);
for (int j = 0; j < size; j++) {
sellHouse.sale();
}
System.out.println(Thread.currentThread().getName()+" 号 "+ sellHouse.saleVolume.get());
}finally {
sellHouse.saleVolume.remove();
downLatch.countDown();
}
},""+i).start();
}
downLatch.await();
System.out.println("main "+sellHouse.saleVolume.get());
}
}
Thread、ThreadLocal、ThreadLocalMap关系
Thread 内 包含属性 ThreadLocal 内 包含静态内部类 ThreadLocalMap
近似的可以理解为:
ThreadLocalMap从字面上就可以看出这是一个保存ThreadLocal对象的map(其实是以ThreadLocal为Key),不过是经过了两层包装的ThreadLocal对象
JVM内存维护了一个线程版的Map<ThreadLocal,Value>(通过ThreadLocal对象的set方法,结果把ThreadLocal对象自己当做key,放进ThreadLocalMap中),每个线程要用到这个T的时候,用当前的线程去Map里面获取,通过这样让每个线程都拥有了自己独立的变量,人手一份,竞争条件被彻底销毁,在并发模式下是绝对安全的变量