基于Logback的MDC实现HTTP请求日志的全链路跟踪

2020-06-02  本文已影响0人  森林中大鸟

原理

final ThreadLocal<Map<String, String>> copyOnThreadLocal = new ThreadLocal();
public void put(String key, String val) throws IllegalArgumentException {
    if (key == null) {
      throw new IllegalArgumentException("key cannot be null");
    } else {
      Map<String, String> oldMap = (Map)this.copyOnThreadLocal.get();
      Integer lastOp = this.getAndSetLastOperation(1);
      if (!this.wasLastOpReadOrNull(lastOp) && oldMap != null) {
        oldMap.put(key, val);
      } else {
        Map<String, String> newMap = this.duplicateAndInsertNewMap(oldMap);
        newMap.put(key, val);
      }

    }
  }

步骤

举例使用过滤器如下

过滤器拦截请求,并添加 traceId到MDC中, 请求处理完毕后,移除MDC中的traceId
MDC中的traceId添加后, 日志输出中可以打印traceId对应的值,每次请求重新生成traceId,确保每个请求traceId不同

package com.xxx.filter;

import java.io.IOException;
import java.util.UUID;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.annotation.WebFilter;
import org.slf4j.MDC;

@WebFilter(urlPatterns = "/*", filterName = "mdcFilter")
public class MdcFilter implements Filter {
    private static final String TRACE_ID = "traceId";

    @Override
    public void init(FilterConfig filterConfig) {
    }

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
        
        try {
            MDC.put(TRACE_ID, UUID.randomUUID().toString().replace("-", ""));
            chain.doFilter(request, response);
        } finally {
            MDC.remove(TRACE_ID);
        }
    }

    @Override
    public void destroy() {
    }
}

logback日志格式修改,添加[%X[traceId]] 到日志输出格式中

<pattern>%d{yy-MM-dd HH:mm:ss.SSS} [%X[traceId]]  [%thread] %-5level %logger{50}:%line - %msg%n</pattern>
上一篇下一篇

猜你喜欢

热点阅读