溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何使用MDC實(shí)現(xiàn)日志鏈路跟蹤

發(fā)布時(shí)間:2021-10-11 09:51:11 來源:億速云 閱讀:171 作者:iii 欄目:開發(fā)技術(shù)

本篇內(nèi)容介紹了“如何使用MDC實(shí)現(xiàn)日志鏈路跟蹤”的有關(guān)知識(shí),在實(shí)際案例的操作過程中,不少人都會(huì)遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!

如何使用MDC實(shí)現(xiàn)日志鏈路跟蹤

  • 前言

  • 原理

  • 實(shí)現(xiàn)

    • 過濾器

    • logback.xml

    • 返回體

  • 效果日志

前言

在微服務(wù)環(huán)境中,我們經(jīng)常使用Skywalking、CAT等去實(shí)現(xiàn)整體請(qǐng)求鏈路的追蹤,但是這個(gè)整體運(yùn)維成本高,架構(gòu)復(fù)雜,我們來使用MDC通過Log來實(shí)現(xiàn)一個(gè)輕量級(jí)的會(huì)話事務(wù)跟蹤功能。

原理

MDC  org.sl4j.MDC其實(shí)內(nèi)部就是ThreadLocal,MDC提供了put/get/clear等幾個(gè)核心接口,用于操作ThreadLocal中的數(shù)據(jù);ThreadLocal中的K-V,可以在logback.xml中聲明,最終將會(huì)打印在日志中。

// java代碼 MDC.put("userId","laker");    // logback.xml %X{userId}

例如:

<property name="pattern" value="%d{HH:mm:ss.SSS} [%thread] %-5level [%X{userId}] %logger{20} - %msg%n"/>

實(shí)現(xiàn)

整體流程如下:

  • 用戶登錄系統(tǒng),我們?nèi)罩局杏涗泆serId:laker。

  • 用戶發(fā)起請(qǐng)求,一個(gè)請(qǐng)求中可能實(shí)際產(chǎn)生多個(gè)http請(qǐng)求,這里可以前端生成一個(gè)requestId

  • 在返回體中,返回requestId。

  • 研發(fā)運(yùn)維人員,可以根據(jù) userId和requestId去日志中撈請(qǐng)求鏈路。

過濾器

@Order(value = Ordered.HIGHEST_PRECEDENCE + 100) @Component @WebFilter(filterName = "MDCFilter", urlPatterns = "/*") public class MDCFilter extends OncePerRequestFilter {      @Override     protected void doFilterInternal(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, FilterChain filterChain) throws ServletException, IOException {         try {             MDC.put("userId", "laker");             MDC.put("requestId", IdUtil.fastUUID());         } catch (Exception e) {             //         }          try {             filterChain.doFilter(httpServletRequest, httpServletResponse);         } finally {             MDC.clear();         }     } }

logback.xml

<?xml version="1.0" encoding="UTF-8"?> <configuration>     <property name="LOG_HOME" value="logs"/>     <property name="encoding" value="UTF-8"/>      <appender name="DEFAULT" class="ch.qos.logback.core.rolling.RollingFileAppender">         <file>${LOG_HOME}/test.log</file>         <Append>true</Append>         <prudent>false</prudent>         <encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder">             <pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%t] %-5level %logger{50} %line - %m%n</pattern>         </encoder>         <!-- 按天回滾 daily -->         <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">             <!--歸檔日志文件名-->             <FileNamePattern>${LOG_HOME}/test.log.%d{yyyy-MM-dd}</FileNamePattern>             <!-- 最多保存15天歷史文件 -->             <maxHistory>15</maxHistory>         </rollingPolicy>     </appender>      <!-- 日志輸出格式 -->     <property name="log.pattern"               value="%d{HH:mm:ss.SSS} [%thread] %-5level [%X{userId}|%X{requestId}] %logger{20} - [%method,%line] - %msg%n"/>      <!-- 控制臺(tái)輸出 -->     <appender name="console" class="ch.qos.logback.core.ConsoleAppender">         <encoder>             <pattern>${log.pattern}</pattern>         </encoder>     </appender>      <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">         <encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder">             <pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%t] %-5level %logger{50} %line - %m%n</pattern>         </encoder>     </appender>      <logger name="com.test.demo" level="DEBUG">         <appender-ref ref="DEFAULT"/>     </logger>     <!-- 日志輸出級(jí)別 -->     <root level="INFO">         <appender-ref ref="DEFAULT"/>         <appender-ref ref="console"/>     </root> </configuration>

返回體

public class Response<T> {     @ApiModelProperty(notes = "響應(yīng)碼,非200 即為異常", example = "200")     private final int code;     @ApiModelProperty(notes = "響應(yīng)消息", example = "提交成功")     private final String msg;     @ApiModelProperty(notes = "響應(yīng)數(shù)據(jù)")     private final T data;     @ApiModelProperty(notes = "請(qǐng)求id")     private final String requestId;      public Response(int code, String msg, T data) {         this.code = code;         this.msg = msg;         this.data = data;         this.requestId = MDC.get("requestId");     }

效果日志

響應(yīng)

{  code: 200,  msg: "",  requestId: "74a269a8-3cb4-417e-853c-b968b77cce23" }

日志

18:37:15.997 [http-nio-8080-exec-1] INFO  [laker|90717490-5ef4-4e46-bc2c-605952fc3803] c.l.m.c.InfoController - [v2Map,17] - null 18:37:38.980 [http-nio-8080-exec-2] INFO  [laker|82bde351-f86e-466f-97a0-c857a0c4c1c9] c.l.m.c.InfoController - [v2Map,17] - null 18:37:39.992 [http-nio-8080-exec-3] INFO  [laker|74a269a8-3cb4-417e-853c-b968b77cce23] c.l.m.c.InfoController - [v2Map,17] - null

“如何使用MDC實(shí)現(xiàn)日志鏈路跟蹤”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí)可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實(shí)用文章!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

mdc
AI