MDC日志追踪

日志追踪

1.如何实现 问题 以及解决方案

初级 每层传参下去 从controller -> service -> dao

中级 通过threalocal

什么是threadlocal 为线程创建一个副本 是以线程号为id 的map 可以塞值 塞我们想要的 traceId

框架MDC

MDC(Mapped Diagnostic Context,映射调试上下文)是 log4j 和 logback 提供的一种方便在多线程条件下记录日志的功能。

MDC中的put方法其实就是讲键值对放入一个Hashtable对象中,然后赋值给当前线程的ThreadLocal.ThreadLocalMap对象,即threadLocals,这保证了各个线程的在MDC键值对的独立性。

边界

1.mdc争对的是当前线程 ,所以当你使用异步,或者说new Thread的时候就不支持了

如何解决: 可以在多线程里面从新设置一遍 MDC.put(“traceId”,“业务唯一键”)

2.分布式的时候 当A系统 调用B系统的时候 下游的系统如何获取traceId

如何解决 从A系统到B系统 走的是hettp 协议 所以我们就可以在url上面做文章

hettp://www.bai.A?traceId=唯一键

springcloud里面的sleuth 就是通过这个方法去实现的

我们直接的服务 怎么去实现他呢 编写两个filter 去实现他

转发的时候对url里面设置一个这个的请求参数

消费端那边先去获取这个traceId 再通过mdc put进去 日志打印的格式加上traceID 这个参数

3.日志采集 服务很多 不知你哪一台去获取 elk 后续再涉及

代码实现

pom文件

<!--一定要先把这里面的logging给排除掉-->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter</artifactId>
    <exclusions>
        <exclusion>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-logging</artifactId>
        </exclusion>
    </exclusions>
</dependency>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-log4j2</artifactId>
</dependency>

application.properties

#指定log的配置文件
logging.config=classpath:log4j2.xml

log4j2.xml

%X{name}会查询设置到MDC中的变量 这个非常重要

%d{yyyy-MM-dd HH:mm:ss,SSS} %t %X{traceId} %-5p %c{1}:%L -%m%n

<?xml version="1.0" encoding="UTF-8"?>
<!--日志级别以及优先级排序: OFF > FATAL > ERROR > WARN > INFO > DEBUG > TRACE > ALL -->
<!-- status log4j2内部日志级别 -->
<configuration status="INFO">
    <!-- 全局参数  %X{name}会查询设置到MDC中的变量 -->
    <Properties>
        <Property name="pattern">%d{yyyy-MM-dd HH:mm:ss,SSS} %t %X{traceId} %-5p %c{1}:%L -%m%n</Property>
        <Property name="displayName">EurekaServer</Property>
    </Properties>
    <Appenders>
        <Console name="console" target="SYSTEM_OUT" follow="true">
            <PatternLayout>
                <pattern>${pattern}</pattern>
            </PatternLayout>
        </Console>
        <!-- 文件 每次运行程序会自动清空,由append属性决定 -->
        <File name="error" fileName="${displayName}_error.log" append="false">
            <!-- 指定error 级别的日志 -->
            <ThresholdFilter level="ERROR" onMatch="ACCEPT"
                             onMismatch="DENY" />
            <PatternLayout>
                <pattern>${pattern}</pattern>
            </PatternLayout>
        </File>
        <!-- 滚动文件 -->
        <RollingFile name="rollingFile" fileName="${displayName}.log"
                     filePattern="${displayName}_%d{yyyy-MM-dd}.log">
            <PatternLayout>
                <pattern>${pattern}</pattern>
            </PatternLayout>
            <!-- 按大小划分 -->
            <SizeBasedTriggeringPolicy size="50 MB" />
        </RollingFile>
    </Appenders>
    <Loggers>
        <!-- <Logger name="org.apache.catalina.util.LifecycleBase" level="ERROR"
            /> <Logger nabaime="org.apache.coyote.http11.Http11NioProtocol" level="WARN"
            /> <Logger name="org.apache.tomcat.util.net.NioSelectorPool" level="WARN"
            /> -->
        <Logger name="org.springframework" level="WARN" />
        <Logger name="com.bai" level="DEBUG" />
        <Root level="DEBUG">
            <AppenderRef ref="console"></AppenderRef>
            <AppenderRef ref="error"></AppenderRef>
            <AppenderRef ref="rollingFile"></AppenderRef>
        </Root>
    </Loggers>
</configuration>

编写任意一个拦截器

import org.slf4j.MDC;
/**
 * @Author BAI
 * @Description 语言转化拦截器
 */
@Slf4j
public class MyLocaleChangeInterceptor implements HandlerInterceptor {
    public static final String TRACE_ID ="traceId";
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws ServletException {
       // 使用mdc塞值  添加日志追踪
        String traceId = UUID.randomUUID().toString().replace("-","");
        MDC.put(TRACE_ID, traceId);
        // 这个添加的操作一定要在preHandle 一般误区会以为会在postHandle
        // 但是使用的时候会发现 在使用postHandle 之前 已经返回给前端
        response.addHeader(TRACE_ID,traceId);
        I18nLocalResolver i18NLocalResolver = new I18nLocalResolver();
        Locale locale = i18NLocalResolver.resolveLocale(request);
        i18NLocalResolver.setLocale(request,response,locale);
        return true;
    }
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        log.info("afterCompletion start...");
        MDC.remove(TRACE_ID);
        log.info("afterCompletion end...");
    }
}

线程池或者异步任务简单处理

String traceId = MDC.get("traceId");
for (int i = 0; i < 1000; i++) {
    newFixedThreadPool(1000).execute(()->{
        MDC.put("traceId",traceId);
        log.info("stu:",stu.getOperateEnum());
        MDC.remove("traceId");
    });
}
上一篇:SpringCloud 中如何微服务只能被指定的程序调用


下一篇:SpringCloud使用traceId跟踪日志解决方案