使用ThreadLocal进行父子线程传值,,传了值也能取到,但是也不总是能取到,若干次之后就总是能取到了,这到底是是为什么呢?

在某个项目中,需要使用mybatis-plus多租户功能以便数据隔离,前端将租户id传到后端,后端通过拦截器将该租户id设置到ThreadLocal以便后续使用,代码大体上如下所示:

  1. ThreadLocal<Integer> threadLocal = new InheritableThreadLocal<>();
  2. threadLocal.set(1);

我在Controller层使用线程池取了租户id,代码大体上如下所示:

  1. ExecutorService executorService = Executors.newSingleThreadExecutor();
  2. executorService.execute(()->{
  3. //获取租户id
  4. });

这时候出问题了,出现了有时候取得到有时候取不到租户id的现象,但是经过若干次重试之后就能稳定获取到租户id;再次测试则发现如果前端传了其它的租户id,后端取得还是上一次获取到的租户id,这到底是为啥呢?

问题分析:首先,这里使用了InheritableThreadLocal为的就是实现父子线程传值,传了值也能取到,但是也不总是能取到,若干次之后就总是能取到了。看到这种现象,我们正常人的第一反应就是怀疑这里有缓存,每次使用的时候没有,使用完了就缓存起来,由于线程池在执行任务的时候并非总是使用同一条线程,当线程池中的核心线程全都缓存完了,再请求就稳定不报错了,然而有缓存的原因所以就算这时候外部请求换了一个租户id,线程池中的线程仍然使用的是老的租户id,这也是缓存最直接的体现。。。。。。这里纯属基于现象的个人猜测,并没有什么实锤,看官们谨慎驾驶,小心翻车。

那怎么解决该问题呢?

该问题产生的原因是InheritableThreadLocal的bug,至于什么bug,我也不清楚(笑),但是有解决方案,解决方案就是使用阿里的transmittable-thread-local 组件,github地址如下:https://github.com/alibaba/transmittable-thread-local 使用起来也非常简单

首先,引入maven依赖:

  1. <dependency>
  2. <groupId>com.alibaba</groupId>
  3. <artifactId>transmittable-thread-local</artifactId>
  4. <version>2.12.0</version>
  5. </dependency>
  1. TransmittableThreadLocal<String> context = new TransmittableThreadLocal<>();
  2. // =====================================================
  3. // 在父线程中设置
  4. context.set("value-set-in-parent");
  5. // =====================================================
  6. // 在子线程中可以读取,值是"value-set-in-parent"
  7. String value = context.get();
  1. ExecutorService executorService = ...
  2. // 额外的处理,生成修饰了的对象executorService
  3. executorService = TtlExecutors.getTtlExecutorService(executorService);

也就是说除了正常创建线程池之外,还要对该线程池做一个代理。

就这么简单,搞完之后父子线程传数据就一切正常了。

ps. 个人觉得这里称呼”父子线程”并不妥当,因为线程池是系统启动之后就已经创建好了的,算了,钻牛角尖太没劲了。

版权声明:本文为kuangdaoyizhimei原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/kuangdaoyizhimei/p/14194234.html