在分布式系统中,链路日志追踪是一项至关重要的功能,可以帮助我们快速定位问题,了解每个请求在系统中的完整调用链路。本文将介绍如何在Spring Boot应用中使用MDC(Mapped Diagnostic Context)实现链路日志追踪,以及如何在使用
@Async
注解的异步任务中传递traceId。
首先,我们需要在Spring Boot项目中引入相关的依赖,并配置日志。接下来,我们将创建一个自定义的拦截器,用于生成traceId并将其添加到MDC中。
在你的Spring Boot项目的pom.xml
文件中,添加以下依赖:
org.springframework.boot
spring-boot-starter-actuator
org.springframework.boot
spring-boot-starter-web
org.slf4j
slf4j-api
1.7.30
新建一个RequestInterceptor
类,实现org.springframework.web.servlet.HandlerInterceptor
接口。在preHandle
方法中,生成一个唯一的ID,将其添加到MDC中。在afterCompletion
方法中,清除MDC的内容。
import org.slf4j.MDC;
import org.springframework.web.servlet.HandlerInterceptor;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.UUID;
public class RequestInterceptor implements HandlerInterceptor {
private static final String TRACE_ID = "traceId";
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {
String traceId = UUID.randomUUID().toString();
MDC.put(TRACE_ID, traceId);
return true;
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) {
MDC.remove(TRACE_ID);
}
}
在Spring Boot的主配置类中,添加一个WebMvcConfigurer
的实现,将RequestInterceptor
注册为一个拦截器。
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
@Configuration
public class WebConfig implements WebMvcConfigurer {
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(new RequestInterceptor());
}
}
在src/main/resources
目录下,创建或编辑logback-spring.xml
文件,将MDC的traceId添加到日志格式中。
%d{HH:mm:ss.SSS} [%X{traceId}] [%thread] %-5level %logger{36} - %msg%n
至此,我们已经实现了基本的链路日志追踪功能。每个请求都会被分配一个唯一的traceId,并在日志中显示。
原先接口响应参数是不包括traceId的,为了方便定位问题,需要在响应的参数中增加。
添加后:
@ApiModel(description = "统一返回结果")
public class Result implements Serializable {
private static final long serialVersionUID = 2872694413533224206L;
@ApiModelProperty(value = "返回编码", example = "200")
private Integer code;
@ApiModelProperty(value = "统一失败信息描述", example = "参数校验失败")
private String message;
@ApiModelProperty(value = "返回的数据")
private T data;
@ApiModelProperty(value = "日志id")
private String logId;
/**
* 自定义:构造结果
*
* @param resultEnum 结果枚举类
* @param data 结果数据
* @return 结果
*/
public static Result build(ResultEnum resultEnum, String message, Object data) {
return new Result(resultEnum.getCode(), message, data,MDC.get(Constants.TRACE_ID));
}
}
当我们在Spring Boot应用中使用@Async
注解来处理异步任务时,由于MDC是基于ThreadLocal实现的,traceId在异步子线程中可能会丢失。为了解决这个问题,我们需要自定义一个AsyncConfigurer
,并使用包含traceId的线程工厂创建线程池。
在你的Spring Boot应用配置类中添加@EnableAsync
注解,启用异步支持:
import org.springframework.scheduling.annotation.EnableAsync;
@Configuration
@EnableAsync
public class AppConfig {
}
Runnable
接口,用于在多线程环境下设置和清除日志跟踪标识(Trace ID)新建一个TraceIdRunnable
类,用于多线程下创建traceId。
构造方法 TraceIdRunnable(Runnable delegate, String traceId)
:接受两个参数,一个是要执行的任务对象 delegate
,另一个是要设置的跟踪标识 traceId
。这些参数被存储在成员变量中。
import org.slf4j.MDC;
public class TraceIdRunnable implements Runnable {
private final Runnable delegate;
private final String traceId;
public TraceIdRunnable(Runnable delegate, String traceId) {
this.delegate = delegate;
this.traceId = traceId;
}
@Override
public void run() {
MDC.put(Constants.TRACE_ID, traceId);
try {
delegate.run();
} finally {
MDC.remove(Constants.TRACE_ID);
}
}
}
新建一个TraceIdThreadFactory
类,用于创建包含traceId的线程。
import org.slf4j.MDC;
import java.util.concurrent.Executors;
import java.util.concurrent.ThreadFactory;
public class TraceIdThreadFactory implements ThreadFactory {
private final ThreadFactory delegate = Executors.defaultThreadFactory();
@Override
public Thread newThread(Runnable r) {
String traceId = MDC.get("traceId");
return delegate.newThread(new TraceIdRunnable(r, traceId));
}
}
新建一个CustomAsyncConfigurer
类,实现AsyncConfigurer
接口。在getAsyncExecutor
方法中,使用TraceIdThreadFactory
创建线程池。
import org.springframework.aop.interceptor.AsyncUncaughtExceptionHandler;
import org.springframework.aop.interceptor.SimpleAsyncUncaughtExceptionHandler;
import org.springframework.scheduling.annotation.AsyncConfigurer;
import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;
import java.util.concurrent.Executor;
@Configuration
@EnableAsync
public class CustomAsyncConfigurer implements AsyncConfigurer {
@Override
public Executor getAsyncExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setCorePoolSize(5);
executor.setMaxPoolSize(10);
executor.setQueueCapacity(25);
executor.setThreadFactory(new TraceIdThreadFactory());
executor.initialize();
return executor;
}
@Override
public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
return new SimpleAsyncUncaughtExceptionHandler();
}
}
现在,异步任务使用@Async
注解创建的子线程将能够访问traceId。这是因为我们通过自定义TraceIdThreadFactory
将traceId从父线程传递给子线程。在子线程的run
方法中,我们将traceId设置到MDC中,并在任务执行完成后清除它。这样,子线程中的日志记录也将包含traceId。
通过本文,我们学习了如何在Spring Boot应用中使用MDC实现链路日志追踪,以及如何在异步任务中传递traceId。这样,我们可以轻松地查看每个请求在系统中的完整调用链路,从而定位问题和优化性能。
以下是我们在本文中完成的主要步骤:
HandlerInterceptor
创建一个请求拦截器,生成traceId并将其添加到MDC中。WebMvcConfigurer
实现中注册请求拦截器。@EnableAsync
注解启用异步支持。ThreadFactory
,用于在异步子线程中传递traceId。AsyncConfigurer
,并使用包含traceId的线程工厂创建线程池。希望这篇文章能帮助你更好地理解Spring Boot中链路日志追踪的实现,以及如何在异步任务中传递traceId。