springboot项目添加请求链路追踪日志traceId

前言:

在springboot项目中,我们想要知道一次请求,从开始到结束的控制台的所有日志,方便快速区分与第二次请求的日志,我们就需要加入traceId链路追踪日志。

注意: 此方法只适用于单项目单线程使用,如果用到多线程如ThreadPoolTaskExecutor(文章末尾有解决方案 ),traceId会断联,因为 MDC 底层是 ThreadLocal ------ 数据只跟当前线程 走。还有在分布式微服务,从A服务发送请求到B服务,请求到达B服务的时候源traceId就会丢失,在B服务就会产生新的traceId。所以你是分布式微服务项目还是建议用SkyWalking/Zipkin等专业的 APM,如果只是简单的开发调试, 这种方式够用了**。**

下面开始上代码,2步搞定:

一,生成并放入 MDC

以下代码可以放入你的项目工具类包中:

复制代码
import cn.hutool.core.lang.UUID;
import org.slf4j.MDC;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;
import org.springframework.web.filter.OncePerRequestFilter;

import javax.servlet.FilterChain;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;


@Component
@Order(-1)   // 保证最先执行
public class TraceIdFilter extends OncePerRequestFilter {

    public static final String TRACE_ID = "traceId";

    @Override
    protected void doFilterInternal(HttpServletRequest request,
                                    HttpServletResponse response,
                                    FilterChain chain) throws ServletException, IOException {
        // 1. 优先取网关传过来的 traceId,没有则新生成
        String traceId = request.getHeader(TRACE_ID);
        if (traceId == null || traceId.isEmpty()) {
            // hutool 无横杠
            traceId = UUID.randomUUID().toString(true);
        }

        // 2. 放入 MDC,后续日志模板里 %X{traceId} 自动打印
        MDC.put(TRACE_ID, traceId);

        // 3. 返回给前端/下游,方便链路追踪
        response.setHeader(TRACE_ID, traceId);

        try {
            chain.doFilter(request, response);
        } finally {
            MDC.clear();   // 防止线程复用串号
        }
    }
}

二,在你的项目日志文件中添加traceId配置

我项目中的日志文件是logback.xml,你看你自己的项目日志文件是那个,将以下配置替换即可

<!-- 原来 -->

<property name="log.pattern"

value="%d{HH:mm:ss.SSS} [%thread] %-5level %logger{20} - [%method,%line] - %msg%n"/>

<!-- 加上 traceId 占位符 -->

<property name="log.pattern"

value="%d{HH:mm:ss.SSS} [%thread] %-5level [%X{traceId}] %logger{20} - [%method,%line] - %msg%n"/>

主要是添加了:[%X{traceId}]

可以先比对一下你日志文件中是否有log.pattern下的%d{HH:mm:ss.SSS} 这一段文本,然后比对一下除**[%X{traceId}]**其他的文本是否一样,如果一样,你就可以直接将以下配置复制粘贴替换就文本:

复制代码
<property name="log.pattern" value="%d{HH:mm:ss.SSS} [%thread] %-5level [%X{traceId}]  %logger{20} - [%method,%line] - %msg%n" />

三,演示效果

重启项目,随便请求一个接口

四,拓展:多线程ThreadPoolTaskExecutor解决traceId断联

复制代码
@Bean
public ThreadPoolTaskExecutor executor() {
    ThreadPoolTaskExecutor exec = new ThreadPoolTaskExecutor();
    exec.setCorePoolSize(8);
    exec.setMaxPoolSize(16);
    exec.setQueueCapacity(200);
    exec.setThreadNamePrefix("async-");
    // ========== 把主线程 MDC 复制到子线程 ==========
    exec.setTaskDecorator(runnable -> {
        Map<String, String> context = MDC.getCopyOfContextMap();   // 快照
        return () -> {
            if (context != null) MDC.setContextMap(context);
            try {
                runnable.run();
            } finally {
                MDC.clear();        // 执行完清掉
            }
        };
    });
    exec.initialize();
    return exec;
}

到此结束!!!

相关推荐
计算机学姐17 分钟前
基于SpringBoot的咖啡店管理系统【个性化推荐+数据可视化统计+配送信息】
java·vue.js·spring boot·后端·mysql·信息可视化·tomcat
My的梦想已实现27 分钟前
关于JAVA Springboot集成支付后打包JAR之后报安全错误的处理
java·spring boot·jar
ooseabiscuit42 分钟前
SpringBoot3整合FastJSON2如何配置configureMessageConverters
java
ok_hahaha1 小时前
java从头开始-黑马点评-Redission
java·开发语言
无巧不成书02181 小时前
Java面向对象零基础实战:从Employee类吃透自定义类核心,掌握封装精髓
java·开发语言·java入门·面向对象·自定义类·employee类·java核心技术
LSTM971 小时前
使用 Python 将图片转换为 PDF (含合并)
后端
小江的记录本1 小时前
【注解】常见 Java 注解系统性知识体系总结(附《全方位对比表》+ 思维导图)
java·前端·spring boot·后端·spring·mybatis·web
小飞Coding1 小时前
MyBatis Mapper 实现原理彻底解密——从动态代理到 JDBC 执行全链路剖析
后端·mybatis
跃上青空1 小时前
Java如何优雅的使用fastjson2进行枚举序列化/反序列化,欢迎探讨
java·开发语言
Mr.45671 小时前
Spring Boot 集成 PostgreSQL 表级备份与恢复实战
java·spring boot·后端·postgresql