服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - ThreadLocal 在上下文传值场景实践源码

ThreadLocal 在上下文传值场景实践源码

2022-09-09 14:07Q.E.D. Java教程

这篇文章主要为大家介绍了ThreadLocal在上下文传值场景下的实践源码,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步

开篇语

我们在 《打动面试官:线程池流程编排中的运用实战》一文中将流程引擎简单地完善了一下,本文在其基础上继续进行改造,建议同学可以先看看 GitHub 上的代码,或者看看之前的文章。

 

1、回顾

流程引擎编排的对象,我们称为组件(就是 SpringBean),之前我们给组件定义了通用的接口,组件实现时就实现这个接口,代码如下:

ThreadLocal 在上下文传值场景实践源码

我们定义了 DomainAbilityBean 接口,入参和出参都是 FlowContent,FlowContent 我们称为上下文。

 

2、ThreadLocal 实现

上下文传参除了 FlowContent 实现外,ThreadLocal 也是可以实现的,我们来演示一下:

2.1、定义 ThreadLocal 上下文工具类

首先我们使用 ThreadLocal 定义了上下文工具类,并且定义了 put、get 方法,方便使用,代码如下:

public class ContextCache implements Serializable {
private static final long serialVersionUID = 2136539028591849277L;
// 使用 ThreadLocal 缓存上下文信息
public static final ThreadLocal<Map<String,String>> CACHE = new ThreadLocal<>();
/**
 * 放数据
 * @param sourceKey
 */
public static final void putAttribute(String sourceKey,String value){
  Map<String,String> cacheMap = CACHE.get();
  if(null == cacheMap){
    cacheMap = new HashMap<>();
  }
  cacheMap.put(sourceKey,value);
  CACHE.set(cacheMap);
}
/**
 * 拿数据
 * @param sourceKey
 */
public static final String getAttribute(String sourceKey){
  Map<String,String> cacheMap = CACHE.get();
  if(null == cacheMap){
    return null;
  }
  return cacheMap.get(sourceKey);
}
}

如果你想往 ThreadLocal 放数据,调用 ContextCache.putAttribute 方法,如果想从 ThreadLocal 拿数据,调用 ContextCache.getAttribute 方法即可。

我们写了两个组件,一个组件放数据,一个组件拿数据,如下:

ThreadLocal 在上下文传值场景实践源码

我们把两个 SpringBean 注册到流程注册中心中,让其按照先执行 BeanThree 再执行 BeanFive 的顺序进行执行,运行 DemoApplication 类的 main 方法进行执行,执行结果如下:

ThreadLocal 在上下文传值场景实践源码

从打印的日志可以看到,在 Spring 容器管理的 SpringBean 中,ThreadLocal 也是可以储存中间缓存值的。

 

3、开启子线程

我们做一个实验,我们在 BeanFive 中开启子线程,然后再从 ThreadLocal 中拿值,看看能否拿到值,BeanFive 的代码修改成如下:

ThreadLocal 在上下文传值场景实践源码

我们再来运行一下,打印的日志如下:

ThreadLocal 在上下文传值场景实践源码

从打印的日志中,我们发现在子线程中从 ThreadLocal 取值时,并没有取得值,这个原因主要是我们之前说的,线程在创建的时候,并不会把父线程的 ThreadLocal 中的值拷贝给子线程的 ThreadLocal,解决方案就是把 ThreadLocal 修改成 InheritableThreadLocal,代码修改如下:

ThreadLocal 在上下文传值场景实践源码

我们再次运行,结果如下:

ThreadLocal 在上下文传值场景实践源码

从运行结果看,我们成功的在子线程中拿到值。

 

4、线程池 + ThreadLocal

如果是拿数据的 springBean 是丢给线程池执行的,我们能够成功的从 ThreadLocal 中拿到数据么?

首先我们在放数据的 springBean 中,把放的值修改成随机的,接着拿数据的 SpringBean 修改成异步执行,代码修改如下:

ThreadLocal 在上下文传值场景实践源码

为了能快速看到效果,我们把线程池的 coreSize 和 maxSize 全部修改成 3,并让任务沉睡一段时间,这样三个线程肯定消费不完任务,大量任务都会到队列中去排队,我们修改一下测试脚本,如下:

ThreadLocal 在上下文传值场景实践源码

我们期望的结果:

线程池中执行的 BeanFive 可以成功从 ThreadLocal 中拿到数据;能够从 ThreadLocal 拿到正确的数据,比如 BeanThree 刚放进 key1,value5,那么期望在 BeanFive 中根据 key1 能拿出 value5,而不是其它值。

我们运行一下,结果如下:

ThreadLocal 在上下文传值场景实践源码

从结果中可以看到,并没有符合我们的预期,我们往 ThreadLocal 中 put 进很多值,但最后拿出来的值却很多都是 value379,都为最后 put 到 ThreadLocal 中的值。

这个原因主要是 ThreadLocal 存储的 HashMap 的引用都是同一个,main 主线程可以修改 HashMap 中的值,子线程从 ThreadLocal 中拿值时,也是从 HashMap 中拿值,从而导致不能把 put 的值通过 ThreadLocal 正确的传递给子线程。

为了证明是这个原因,我们在从 ThreadLocal 放、拿值的地方,把 HashMap 的内存地址都打印出来,改动代码如下:

ThreadLocal 在上下文传值场景实践源码

我们再次运行测试代码,运行的结果如下:

ThreadLocal 在上下文传值场景实践源码

从测试结果中可以看到,不管是主线程还是子线程和 ThreadLocal 进行交互时,HashMap 都是同一个,也就是说 ThreadLocal 中保存的 HashMap 是共享的,这就导致了线程安全的问题,子线程读取到的值就会混乱掉。

 

5、解决方案

针对这个问题,我们提出了一种解决方案,在把任务提交到线程池时,我们进行 HashMap 的拷贝,这样子线程的 HashMap 和 main 线程的 HashMap 就不同了,可以解决上面的问题。

我们提交任务时, 使用的是 Runnable,要实现 HashMap 的拷贝的话,我们需要把 Runnable 进行一层包装,包装的代码如下:

ThreadLocal 在上下文传值场景实践源码

运行结果如下:

ThreadLocal 在上下文传值场景实践源码

从运行结果中可以看出,线程池拿出来的 value 已经是正确的了。

 

6、总结

本文通过 ThreadLocal 来改造流程引擎中的上下文传递,希望能够加深大家对 ThreadLocal 的认识和使用技巧,有兴趣的同学可以把我们的代码下载下来,跑跑看。

以上就是ThreadLocal 在上下文传值场景下的实践的详细内容,更多关于ThreadLocal 上下文传值场景实践的资料请关注服务器之家其它相关文章!

原文链接:https://blog.csdn.net/qq_34272760/article/details/120661903

延伸 · 阅读

精彩推荐
  • Java教程Java中List集合去重方法以及效率对比

    Java中List集合去重方法以及效率对比

    这篇文章主要给大家介绍了关于Java中List集合去重方法以及效率对比的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参...

    格姗知识圈4612021-09-01
  • Java教程SpringMVC实现文件上传和下载功能

    SpringMVC实现文件上传和下载功能

    这篇文章主要为大家详细介绍了SpringMVC实现文件上传和下载功能 ,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    AltErNatiVe3272020-12-21
  • Java教程lombok插件无法使用的原因及解决方案

    lombok插件无法使用的原因及解决方案

    这篇文章主要介绍了lombok插件无法使用的原因及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教...

    熊熊爱绵羊12192021-11-03
  • Java教程通过Java来测试JSON和Protocol Buffer的传输文件大小

    通过Java来测试JSON和Protocol Buffer的传输文件大小

    这篇文章主要介绍了通过Java来测试JSON和Protocol Buffer的传输文件大小,Protocol Buffer(文中简称Protobuffer)是谷歌开发的新的文件传输格式,需要的朋友可以参考下...

    cxshun4272020-03-14
  • Java教程Spring MVC 拦截器实现登录

    Spring MVC 拦截器实现登录

    这篇文章主要介绍了Spring MVC 拦截器实现登录,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    钱春华4672020-11-27
  • Java教程idea关联maven的使用详解

    idea关联maven的使用详解

    这篇文章主要介绍了idea关联maven的使用详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考...

    你看星星很亮6232021-08-30
  • Java教程Spring Boot+maven打war包的方法

    Spring Boot+maven打war包的方法

    这篇文章主要介绍了Spring Boot+maven打war包的方法,本文通过实例代码相结合的形式给大家介绍的非常详细,需要的朋友参考下吧...

    woncode5582021-04-25
  • Java教程JAVA实现基于皮尔逊相关系数的相似度详解

    JAVA实现基于皮尔逊相关系数的相似度详解

    这篇文章主要介绍了JAVA实现基于皮尔逊相关系数的相似度详解,具有一定参考价值,需要的朋友可以了解下。...

    panjiao1199962021-02-22