十三、ThreadLocal、InheritableThreadLocal
1、ThreadLocal简介
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-qYHi3f2h-1667184891313)(images/1.JUC并发编程/image-20210927222129464.png)]
稍微翻译一下:
ThreadLocal提供线程局部变量。这些变量与正常的变量不同,因为每一个线程在访问ThreadLocal实例的时候(通过其get或set方法)都有自己的、独立初始化的变量副本。ThreadLocal实例通常是类中的私有静态字段,使用它的目的是希望将状态(例如,用户ID或事务ID)与线程关联起来。
实现每一个线程都有自己专属的本地变量副本(自己用自己的变量不麻烦别人,不和其他人共享,人人有份,人各一份),主要解决了让每个线程绑定自己的值,通过使用get()和set()方法,获取默认值或将其值更改为当前线程所存的副本的值从而避免了线程安全问题。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-RZbRldkH-1667184891313)(images/1.JUC并发编程/image-20210927222231797.png)]
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-yaczweH1-1667184891313)(images/1.JUC并发编程/image-20210927222241804.png)]
2、永远的helloworld
按照总销售额统计,方便集团公司做计划统计
class MovieTicket
{
int number = 50;
public synchronized void saleTicket()
{
if(number > 0)
{
System.out.println(Thread.currentThread().getName()+"\t"+"号售票员卖出第: "+(number--));
}else{
System.out.println("--------卖完了");
}
}
}
/**
* 三个售票员卖完50张票务,总量完成即可,吃大锅饭,售票员每个月固定月薪
*/
public class ThreadLocalDemo
{
public static void main(String[] args)
{
MovieTicket movieTicket = new MovieTicket();
for (int i = 1; i <=3; i++) {
new Thread(() -> {
for (int j = 0; j <20; j++) {
movieTicket.saleTicket();
try { TimeUnit.MILLISECONDS.sleep(10); } catch (InterruptedException e) { e.printStackTrace(); }
}
},String.valueOf(i)).start();
}
}
}
不参加总和计算,希望各自分灶吃饭,各凭销售本事提成,按照出单数各自统计
class MovieTicket
{
int number = 50;
public synchronized void saleTicket()
{
if(number > 0)
{
System.out.println(Thread.currentThread().getName()+"\t"+"号售票员卖出第: "+(number--));
}else{
System.out.println("--------卖完了");
}
}
}
class House
{
ThreadLocal<Integer> threadLocal = ThreadLocal.withInitial(() -> 0);
public void saleHouse()
{
Integer value = threadLocal.get();
value++;
threadLocal.set(value);
}
}