欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時間:2022年01月17日 09:25:56   作者:lakernote?  
這篇文章主要介紹了使用MDC實現(xiàn)日志鏈路跟蹤,在微服務(wù)環(huán)境中,我們經(jīng)常使用Skywalking、CAT等去實現(xiàn)整體請求鏈路的追蹤,但是這個整體運維成本高,架構(gòu)復(fù)雜,我們來使用MDC通過Log來實現(xiàn)一個輕量級的會話事務(wù)跟蹤功能,下面就來看看具體的過程吧,需要的朋友可以參考一下

前言:

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

1.原理

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

// 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.實現(xiàn)

整體流程如下:

  • 用戶登錄系統(tǒng),我們?nèi)罩局杏涗?code>userId:laker。
  • 用戶發(fā)起請求,一個請求中可能實際產(chǎn)生多個http請求,這里可以前端生成一個requestId
  • 在返回體中,返回requestId。
  • 研發(fā)運維人員,可以根據(jù) userIdrequestId去日志中撈請求鏈路。

3.過濾器

@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"/>

? ? <!-- 控制臺輸出 -->
? ? <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>
? ? <!-- 日志輸出級別 -->
? ? <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 = "請求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實現(xiàn)日志鏈路跟蹤的文章就介紹到這了,更多相關(guān)MDC實現(xiàn)日志鏈路跟蹤內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • springboot項目之相互依賴報錯問題(基于idea)

    springboot項目之相互依賴報錯問題(基于idea)

    這篇文章主要介紹了springboot項目之相互依賴報錯問題(基于idea),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-02-02
  • springmvc實現(xiàn)簡單的攔截器

    springmvc實現(xiàn)簡單的攔截器

    這篇文章主要為大家詳細(xì)介紹了springmvc實現(xiàn)簡單攔截器的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-05-05
  • java實現(xiàn)一個桌球小游戲

    java實現(xiàn)一個桌球小游戲

    這篇文章主要為大家詳細(xì)介紹了java實現(xiàn)一個桌球小游戲,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-07-07
  • 簡單了解Java創(chuàng)建線程兩種方法

    簡單了解Java創(chuàng)建線程兩種方法

    這篇文章主要介紹了簡單了解Java創(chuàng)建線程兩種方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-02-02
  • Java中ThreadLocal線程變量的實現(xiàn)原理

    Java中ThreadLocal線程變量的實現(xiàn)原理

    本文主要介紹了Java中ThreadLocal線程變量的實現(xiàn)原理,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-06-06
  • JAVA-4NIO之Channel之間的數(shù)據(jù)傳輸方法

    JAVA-4NIO之Channel之間的數(shù)據(jù)傳輸方法

    下面小編就為大家?guī)硪黄狫AVA-4NIO之Channel之間的數(shù)據(jù)傳輸方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-06-06
  • java實現(xiàn)打字游戲小程序

    java實現(xiàn)打字游戲小程序

    這篇文章主要為大家詳細(xì)介紹了java實現(xiàn)打字游戲小程序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-10-10
  • 快速理解spring中的各種注解

    快速理解spring中的各種注解

    這篇文章主要介紹了快速理解spring中的各種注解,具有一定借鑒價值,需要的朋友可以了解下。
    2017-12-12
  • 關(guān)于Java中代碼塊的執(zhí)行順序

    關(guān)于Java中代碼塊的執(zhí)行順序

    這篇文章主要介紹了關(guān)于Java中代碼塊的執(zhí)行順序,構(gòu)造代碼塊是給所有對象進(jìn)行統(tǒng)一初始化,而構(gòu)造函數(shù)是給對應(yīng)的對象初始化,因為構(gòu)造函數(shù)是可以多個的,運行哪個構(gòu)造函數(shù)就會建立什么樣的對象,但無論建立哪個對象,都會先執(zhí)行相同的構(gòu)造代碼塊,需要的朋友可以參考下
    2023-08-08
  • Java實現(xiàn)生成自定義時長的靜音音頻

    Java實現(xiàn)生成自定義時長的靜音音頻

    這篇文章主要介紹了如何通過Java實現(xiàn)一個音頻工具類,可以實現(xiàn)生成一段自定義時長(精確到毫秒)的wav音頻。感興趣的小伙伴可以了解一下
    2022-01-01

最新評論