SpringBoot+slf4j线程池全链路调用日志跟踪问题及解决思路(二)

网友投稿 1303 2023-01-21

SpringBoot+slf4j线程池全链路调用日志跟踪问题及解决思路(二)

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 Future submit(Runnable task, T result) {

return super.submit(ThreadMdcUtil.wrap(task, MDC.getCopyOfContextMap()), result);

}

@Override

public Future submit(Callable task) {

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 Callable wrap(final Callable callable, final Map context) {

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小时内删除侵权内容。

上一篇:app应用分发(app应用分发平台源码)
下一篇:vue小程序(vue小程序模板开发)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~