对于常见的WEB容器,Takin通过加强org.apache.catalina.core.StandardHostValve#invoke办法,拦挡并解析办法入参的Request对象中的header数据,判断压测标的信息并将压测标以及链路信息设置到上下文对象,并通过ThreadLocal保留该上下文对象,以此来保障以后线程解决的过程中都能获取到压测标以及链路信息。

然而不论是WEB容器还是业务代码常常会应用线程池运行代码,这时候ThreadLocal就有局限性了,如何做到跨线程池传递咱们的上下文对象就显得异样重要了,好在Alibaba的https://github.com/alibaba/tr...解决了这个问题,具体transmittable-thread-local的具体内容大家能够通过git地址查看,这里咱们次要以git上的demo看看transmittable-thread-local如何实现跨线程传递的,次要通过将线程池运行的Runnable对象包装成TtlRunnable对象并保留以后的上下文信息。这样就实现了对线程池运行工作的压测标以及链路数据信息的透传,至此仿佛所有都很完满。



直到某次呈现了一个诡异的景象,在应用Takin的过程中,有一利用接入Takin之后,某个性能无奈失常应用,然而无论是利用的日志还是Takin的日志都未显示任何异样信息,再诘问了无奈应用的性能是一个线程池执行工作,过后扩大了ThreadPoolExecutor的beforeExecute办法和afterExecute办法,用以记录工作执行的耗时,并且会打印工作的taskId等信息,通过截图的Debug信息便能够看到问题呈现了,因为应用了transmittable-thread-local导致原始的Runnable被包装成了TtlRunnable对象,在这个过程中如果业务想要再获取Runnable对象强转获取一些工作信息,就会呈现类无奈转换的异样。

解决方案的话目前能够通过加强beforeExecute和afterExecute办法,对Runnable对象进行判断,如果是TtlRunnable对象,则通过TtlRunnable.getRunnable()办法返回原是的Runnable对象


Takin开源详情https://news.shulie.io/?p=3024
立刻扫码,入群技术交换~