up:: ThreadLocal第一种使用场景

说明:

(1) 本篇博客内容:介绍ThreadLocal的第二种使用场景;

(2) ThreadLocal第二种使用场景,通常会用在:要完成某个业务需要调用很多方法,同时这个业务中有个信息需要在各个方法中都会使用的到;那么对于这个各个方法都会使用到的信息,就可以使用ThreadLocal保存起来;那么,在需要使用这个信息的地方,我们直接通过ThreadLocal获取就行了,而不用再通过层层传递参数的方式了

(3) 声明:自己在实际工作中,并没有做过并发的web项目;所以,本篇博客的一些想法、观点,可能存在不足甚至错误的地方;所以,尽快的了解一下并发的web项目,还是挺有必要的;


一:ThreadLocal第二种使用场景,场景介绍;

1.场景介绍;

(1) 比如这个案例;

● 假设有个web项目,其中有个业务线是:用户下单;而这个业务,需要包含几个功能:【首先,获取当前登录用户的User信息】,【然后,显示欢迎该用户的提示信息】,【然后,还有就是,显示隶属于该用户的优惠券】,【然后,就是该用户下单】;

● 可以看到,【获取User】中获取的用户User信息,在【显示欢迎信息】【优惠券】【下单】都是需要用到的;

● 通过以往的经验,为了应对这个需求,我们有两种解决方案:

传统解决方案一: 最直接的想法是:【获取User】获取到User信息后,通过调用【显示欢迎信息】【优惠券】【下单】中的方法,通过方法参数的形式把User信息传给【显示欢迎信息】【优惠券】【下单】;:这种方式,太笨重了,如果有一个业务有十几个Service的话,那么我们在调用方法的时候,每次都要附带User,这不仅会导致代码冗余,而且我们每次还要判断一下上层传过来的User是否为空;

传统解决方案二: 后来,我们使用了过滤器,通过过滤器拦截请求,获取User;然后,在需要User的地方,直接通过过滤器类来获取User就行了;可以参考【SpringBoot电商项目购物车模块统一校验当前是否有用户登录】;;可以发现,这种方式的本质是【每一个任务来的时候,这个任务中会有一个信息,同时为了完成这个任务需要调用很多方法,而这些方法大都需要使用到这个信息;;;为此,我们定义一个专门存放该信息的类,这个类会从任务中提取出信息,然后把其存放在 该类的 变量中;;;;这样一来,在后续处理这个任务的方法中,如果需要这个信息,那么直接通过这个类就能获取的到】;;;(首先声明一下,自己并没有在工作中做过并发的web项目,所以以下内容纯属胡扯)但是,一旦有并发需求的时候,我们需要为每个线程创建各自的【专门存放该信息的类的对象】,才能防止线程安全问题;

● 声明:眼光不要太狭隘,这儿仅仅是以【获取User】为例;这种思想完全是可以泛华的;只要符合这个逻辑,任何信息都是可以的;

● 其实,对于上面的那种情况,使用ThreadLocal可以更优雅的解决;这也是ThreadLocal的第二种应用场景;

(2) ThreadLocal第二种使用场景:每个Thread线程类中,都有自己的存储任务信息的对象,同时,各个线程之间的对象是不共享的;

● 通俗的来讲,第二种场景就是: 【每个任务都有专属于该任务的一个信息(比如请求中的User信息)】→【然后处理这个任务需要用到很多方法(比如处理一个用户下单请求,就可能需要【显示欢迎信息】【优惠券】【下单】等不同的Service)】→【然后在这些方法中都需要用到这个信息(比如【显示欢迎信息】【优惠券】【下单】,都需要用到User信息)】→【很显然,处理这个任务的逻辑可能面临多线程并发的需求(比如,同一时间有很多用户一起下单,那么下单逻辑就需要处理并发请求)】→【为了能防止多线程并发处理不同请求任务的时候,每个处理请求任务的线程在调用对应的方法时,获取的User信息都是当前线程中的、那个请求的、User信息】;;;;;为了能够更高效、优雅的解决这个需求,ThreadLocal第二种使用场景就可以比较好的应对;

● 个人感觉,思路不要局限在web请求中;任何符合这种场景的逻辑,似乎都可使用ThreadLocal的第二种场景去应对;

2.演进分析;

还是上面的那个需求场景:

(1) 我们可不可以把任务中的user信息,存储到一个user map中,然后在需要user信息的时候,直接从这个map中获取;(PS:这种方案的本质和上面的【传统解决方案二】是一样的);

但是,如果多个任务过来,我们多线程并发执行的时候,因为map不是线程安全的,所以也会导致线程安全问题;所以,此时要么我么给map加锁,要么我们就使用一种线程安全的map:ConcurrentHashMap;;;但,无论怎么做,这都会影响性能;

(2) ThreadLocal第二种使用场景:

● 就可以比较好的解决这个痛点;在多线程并发执行任务的时候,其既不用像【 传统解决方案一】那样层层传递参数来做;又能保证在多线程执行任务的时候,该线程中持有的信息就是当前所执行任务的信息;还能够,线程在执行某个任务的时候,可以随时随地的获取到任务信息;

● ThreadLocal的第二种使用场景,强调的是:同一个请求内(同一个线程内)不同方法之间的共享;


二:ThreadLocal第二种使用场景,演示;

1.创建User类;

2.创建UserContextHolder类(ThreadLocal工具类):用来存放【当前线程所运行任务的,user信息】;

 
     package threadLocal;
 
     public class UserContextHolder {
         public static ThreadLocal<User> holder = new ThreadLocal<>();
     }

说明:

(1) 此时,我们创建的ThreadLocal,就相当于是一个存储的容器;当有信息被存进来的时,这个容器能够识别是哪个线程中的任务存的;然后,当某个线程中的任务过来取信息的时候,这个容器能够找到 该线程 存的那个信息;

3.然后创建几个Service,以模拟;

4.创建ThreadLocalNormalUsage06类,去模拟发起100次请求;

 
     package threadLocal;
 
     import java.util.concurrent.ExecutorService;
     import java.util.concurrent.Executors;
 
     public class ThreadLocalNormalUsage06 {
 
         public static void main(String[] args) {
             ExecutorService executorService = Executors.newFixedThreadPool(10);
 
 
             for (int i = 0; i <100 ; i++) {
                 String finalI = i + "";
                 executorService.submit(new Runnable() {
                     @Override
                     public void run() {
                         new Service1().process1(finalI);
                     }
                 });
             }
             executorService.shutdown();
         }
     }
 

说明:

(1) 类内容说明;

(2) 运行结果;