使用MDC實(shí)現(xiàn)日志鏈路跟蹤
前言:
在微服務(wù)環(huán)境中,我們經(jīng)常使用Skywalking、CAT等去實(shí)現(xiàn)整體請(qǐng)求鏈路的追蹤,但是這個(gè)整體運(yùn)維成本高,架構(gòu)復(fù)雜,我們來(lái)使用MDC通過(guò)Log來(lái)實(shí)現(xiàn)一個(gè)輕量級(jí)的會(huì)話事務(wù)跟蹤功能。
1.原理
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"/>2.實(shí)現(xiàn)
整體流程如下:
- 用戶登錄系統(tǒng),我們?nèi)罩局杏涗?code>userId: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)求鏈路。
3.過(guò)濾器
@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();
? ? ? ? }
? ? }
}4.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>5.返回體
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");
? ? }6.效果日志
響應(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
到此這篇關(guān)于使用MDC實(shí)現(xiàn)日志鏈路跟蹤的文章就介紹到這了,更多相關(guān)MDC實(shí)現(xiàn)日志鏈路跟蹤內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
springboot使用@Slf4j進(jìn)行日志的記錄步驟詳解
這篇文章主要介紹了springboot使用@Slf4j進(jìn)行日志的記錄,使用@Slf4j的注解進(jìn)行日志記錄非常方便,本文給大家分享操作步驟,需要的朋友可以參考下2023-08-08
基于Java SSM的健康管理小程序的實(shí)現(xiàn)
本篇文章主要為大家分享了基于SSM健康管理小程序的設(shè)計(jì)與實(shí)現(xiàn)。感興趣的小伙伴可以了解一下2021-11-11
Java實(shí)現(xiàn)的決策樹(shù)算法完整實(shí)例
這篇文章主要介紹了Java實(shí)現(xiàn)的決策樹(shù)算法,簡(jiǎn)單描述了決策樹(shù)的概念、原理,并結(jié)合完整實(shí)例形式分析了java實(shí)現(xiàn)決策樹(shù)算法的相關(guān)操作技巧,代碼中備有較為詳盡的注釋便于理解,需要的朋友可以參考下2017-11-11
Spring之AOP兩種代理機(jī)制對(duì)比分析(JDK和CGLib動(dòng)態(tài)代理)
這篇文章主要介紹了Spring之AOP兩種代理機(jī)制對(duì)比分析(JDK和CGLib動(dòng)態(tài)代理),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-05-05
Java基礎(chǔ)類庫(kù)之StringBuffer類用法詳解
String類是在所有開(kāi)發(fā)項(xiàng)目開(kāi)發(fā)之中一定會(huì)使用的一個(gè)功能類。雖然String類很好用,但也有弊端——內(nèi)容不允許頻繁修改,所以為了解決問(wèn)題,我們提供了StringBuffer類。本文就來(lái)講講StringBuffer類的用法2022-07-07
解決java連接虛擬機(jī)Hbase無(wú)反應(yīng)的問(wèn)題
這篇文章主要介紹了解決java連接虛擬機(jī)Hbase無(wú)反應(yīng)的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-06-06

