react 前端框架如何驱动企业数字化转型与创新发展
1303
2023-01-21
SpringBoot+slf4j线程池全链路调用日志跟踪问题及解决思路(二)
本项目源码已在多个项目中实践
接着上一篇文章,项目中使用了线程池,那么子线程中日志就会丢失traceId,下面讲解如何实现子线程中的traceId日志跟踪。
解决思路
子线程在打印日志的过程中traceId将丢失,解决方式为重写线程池,将主线程的traceId继续传递到子线程中。当然,对于直接new创建线程的情况不考略【实际应用中应该避免这种用法】。
继承ThreadPoolExecutor,重写执行任务的方法
public final class OverrideThreadPoolExecutor extends ThreadPoolExecutor {
@Override
public void execute(Runnable task) {
super.execute(ThreadMdcUtil.wrap(task, MDC.getCopyOfContextMap()));
}
@Override
public
return super.submit(ThreadMdcUtil.wrap(task, MDC.getCopyOfContextMap()), result);
}
@Override
public
return super.submit(ThreadMdcUtil.wrap(task, MDC.getCopyOfContextMap()));
}
@Override
public Fhttp://uture> submit(Runnable task) {
return super.submit(ThreadMdcUtil.wrap(task, MDC.getCopyOfContextMap()));
}
}
封装ThreadMdcUtil工具类
以封装Cal
http://
lable为例:
判断当前线程对应MDC的Map是否存在,如果存在则设置子线程的ContextMap为当前线程的;
如果不存在,则重新生成traceId;
执行run方法
public final class ThreadMdcUtil {
public static void setThttp://raceIdIfAbsent() {
if (MDC.get(TraceConstant.MDC_TRACE) == null || MDC.get(TraceConstant.MDC_TRACE).length() == 0) {
String tid = UUID.randomUUID().toString().replace("-", "");
MDC.put(TraceConstant.MDC_TRACE, tid);
}
}
public static
return () -> {
if (context == null) {
MDC.clear();
} else {
MDC.setContextMap(context);
}
setTraceIdIfAbsent();
try {
return callable.call();
} finally {
MDC.clear();
}
};
}
}
测试子线程中traceId的传递,本项目中ExecutorService已经重写了线程池
@RestController
@RequestMapping("trace")
@Slf4j
@AllArgsConstructor
public class TestTraceController {
private final ExecutorService executorService;
@GetMapping("traceLog")
public String traceLog() {
log.info("---接口调用了---");
traceService();
asyncTrace();
return "success";
}
private void traceService(){
log.error("## 执行traceService方法");
}
private void asyncTrace(){
CompletableFuture.runAsync(()->{
log.info("执行线程池中的方法asyncTrace,未重写了线程池");
}, executorService);
}
}
未重写ThreadPoolExecutor效果如下:
重写ThreadPoolExecutor效果如下:
版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。
发表评论
暂时没有评论,来抢沙发吧~