有了InheritableThreadLocal为啥还需要TransmittableThreadLocal?

有了InheritableThreadLocal为啥还需要TransmittableThreadLocal?

典型回答

InheritableThreadLocal是用于主子线程之间参数传递的,但是,这种方式有一个问题,那就是必须要是在主线程中手动创建的子线程才可以,而现在池化技术非常普遍了,很多时候线程都是通过线程池进行创建和复用的,这时候InheritableThreadLocal就不行了。 TransmittableThreadLocal是阿里开源的一个方案 (开源地址) ,这个类继承并加强InheritableThreadLocal类。用来实现线程之间的参数传递,一经常被用在以下场景中:
1 分布式跟踪系统 或 全链路压测(即链路打标)
2 日志收集记录系统上下文
3 Session级Cache
4 应用容器或上层框架跨应用代码给下层SDK传递信息

使用方式

先需要导入依赖:

<dependency> 
  <groupId>com.alibaba</groupId> 
  <artifactId>transmittable-thread-local</artifactId>
  <version>2.14.2</version>
</dependency>

对于简单的父子线程之间参数传递,可以用以下方式:

TransmittableThreadLocal<String> context = new TransmittableThreadLocal<>();
// 在父线程中设置 
context.set("value-set-in-parent"); 
// 在子线程中可以读取,值是"value-set-in-parent" 
String value = context.get();

如果在线程池中,可以用如下方式使用:

TransmittableThreadLocal<String> context = new TransmittableThreadLocal<>(); // 在父线程中设置
context.set("value-set-in-parent");
Runnable task = new RunnableTask();

// 额外的处理,生成修饰了的对象
ttlRunnable Runnable ttlRunnable = TtlRunnable.get(task);
executorService.submit(ttlRunnable);

// Task中可以读取,值是"value-set-in-parent"
String value = context.get();

除了Runnable,Callable也支持:

TransmittableThreadLocal<String> context = new TransmittableThreadLocal<>();
// 在父线程中设置 context.set("value-set-in-parent");
Callable call = new CallableTask();
// 额外的处理,生成修饰了的对象
ttlCallable Callable ttlCallable = TtlCallable.get(call);
executorService.submit(ttlCallable);
// Call中可以读取,值是"value-set-in-parent"

也可以直接用在线程池上,而不是Runnable和Callable上:

ExecutorService executorService = ...
// 额外的处理,生成修饰了的对象
executorService executorService = TtlExecutors.getTtlExecutorService(executorService); 
TransmittableThreadLocal<String> context = new TransmittableThreadLocal<>(); 
// 在父线程中设置 context.set("value-set-in-parent");
Runnable task = new RunnableTask();
Callable call = new CallableTask();
executorService.submit(task);
executorService.submit(call);
// Task或是Call中可以读取,值是"value-set-in-parent" String value = context.get();
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
TransmittableThreadLocalInheritableThreadLocal都是用于解决线程间上下文传递的问题,但是它们的实现方式略有不同。 TransmittableThreadLocal是可传输的线程局部变量,它可以在使用线程池等会池化复用线程的执行组件情况下,提供ThreadLocal值的传递功能。也就是说,当一个线程从线程池中取出来执行任务时,它可以获取到之前设置的TransmittableThreadLocal的值,从而保证了上下文的传递。TransmittableThreadLocal继承自InheritableThreadLocal。 而InheritableThreadLocal是可继承的线程局部变量,它用于父线程传递本地变量到子线程。当一个线程创建子线程时,子线程可以获取到父线程设置的InheritableThreadLocal的值,从而保证了上下文的传递。InheritableThreadLocal继承自ThreadLocal,并且重写了父类的方法:childValue、getMap、createMap。 因此,TransmittableThreadLocalInheritableThreadLocal的区别在于它们的应用场景和实现方式不同。 代码演示如下: ```python import threading from threading import Thread, current_thread from concurrent.futures import ThreadPoolExecutor from threading import Lock from concurrent.futures import Future from threading import Event from threading import Semaphore from threading import Barrier from threading import Condition from threading import Timer from threading import ThreadLocal, local from concurrent.futures import as_completed from concurrent.futures import wait from concurrent.futures import FIRST_COMPLETED from concurrent.futures import ALL_COMPLETED from collections import deque from threading import InheritableThreadLocal, current_thread from threading import Thread # InheritableThreadLocal class MyLocal(InheritableThreadLocal): def __init__(self, value): self.value = value def __repr__(self): return str(self.value) local_data = MyLocal(1) def worker(): print(current_thread().getName(), 'starts with', local_data) local_data.value += 1 print(current_thread().getName(), 'ends with', local_data) t = Thread(target=worker) t.start() t.join() # TransmittableThreadLocal from pytransmit import TransmitThreadLocal transmit_data = TransmitThreadLocal() def worker(): print(current_thread().getName(), 'starts with', transmit_data.get()) transmit_data.set(2) print(current_thread().getName(), 'ends with', transmit_data.get()) t = Thread(target=worker) t.start() t.join() ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值