SpringBoot實(shí)現(xiàn)簡單的日志鏈路追蹤
一. 背景
隨著分布式應(yīng)用的普及,現(xiàn)在的一些應(yīng)用系統(tǒng)不再像以前,所有的文件(前后端程序)都打包在一個(gè)包中,現(xiàn)在的很多應(yīng)用都是模塊化開發(fā),開發(fā)的團(tuán)隊(duì)也是不一樣,服務(wù)與服務(wù)之間的調(diào)用也比較多,在這種情況下,系統(tǒng)的日志就顯得尤其的重要,然而,在多數(shù)情況下,當(dāng)我們的系統(tǒng)出現(xiàn)了異常,需要查看日志時(shí),就會(huì)很抓狂。為了避免這種情況,我們需要把同一次的業(yè)務(wù)調(diào)用鏈上的日志串聯(lián)起來。
本次通過一個(gè)簡單的SpringBoot應(yīng)用來總結(jié),我們?nèi)绾螌⑷罩敬?lián)起來,以下截圖是最終的實(shí)現(xiàn)效果

二. 代碼演示
1. 創(chuàng)建一個(gè)SpringBoot項(xiàng)目
使用idea創(chuàng)建一個(gè)
SpringBoot項(xiàng)目的詳細(xì)步驟,本文不介紹了,具體的步驟,網(wǎng)上有很多例子可以參考
1.1. 我的SpringBoot項(xiàng)目名稱是:springboot-track,以下是工程pom.xml文件中所需要的必要依賴
<dependencies> ?<dependency> ? ?<groupId>org.springframework.boot</groupId> ? ?<artifactId>spring-boot-starter-web</artifactId> ?</dependency> ?<dependency> ? ?<groupId>org.springframework.boot</groupId> ? ?<artifactId>spring-boot-starter-logging</artifactId> ?</dependency> ? ?<dependency> ? ?<groupId>org.projectlombok</groupId> ? ?<artifactId>lombok</artifactId> ? ?<optional>true</optional> ?</dependency> ?<dependency> ? ?<groupId>org.springframework.boot</groupId> ? ?<artifactId>spring-boot-starter-test</artifactId> ? ?<scope>test</scope> ?</dependency> </dependencies>
1.2. 在項(xiàng)目的resource目錄下,創(chuàng)建日志框架整合配置文件:logback-spring.xml文件內(nèi)容配置如下:
<?xml version="1.0" encoding="UTF-8"?>
<configuration debug="false">
?<!--日志存儲(chǔ)路徑-->
?<property name="log" value="/Users/username/Downloads"/>
?<!-- 控制臺(tái)輸出 -->
?<appender name="console" class="ch.qos.logback.core.ConsoleAppender">
? ?<encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder">
? ? ?<!--輸出格式化-->
? ? ?<pattern>[%X{TRACE_ID}] %d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50} - %msg%n</pattern>
? ?</encoder>
?</appender>
?<!-- 按天生成日志文件 -->
?<appender name="file" class="ch.qos.logback.core.rolling.RollingFileAppender">
? ?<rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
? ? ?<!--日志文件名-->
? ? ?<FileNamePattern>${log}/%d{yyyy-MM-dd}.log</FileNamePattern>
? ? ?<!--保留天數(shù)-->
? ? ?<MaxHistory>30</MaxHistory>
? ?</rollingPolicy>
? ?<encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder">
? ? ?<pattern>[%X{TRACE_ID}] %d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50} - %msg%n</pattern>
? ?</encoder>
? ?<!--日志文件最大的大小-->
? ?<triggeringPolicy class="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy">
? ? ?<MaxFileSize>10MB</MaxFileSize>
? ?</triggeringPolicy>
?</appender>
?
?<!-- 日志輸出級(jí)別 -->
?<root level="INFO">
? ?<appender-ref ref="console"/>
? ?<appender-ref ref="file"/>
?</root>
</configuration>1.3. 在項(xiàng)目的resource目錄下的主配置文件(application.yml)中添加日志整合配置信息,添加內(nèi)容如下:
需要注意的是:使用
idea創(chuàng)建的SpringBoot項(xiàng)目,application文件的默認(rèn)后綴是.properties,本人比較喜歡.yml文件,所以將文件后綴名修改了一下
server: port: 8080 logging: config: classpath:logback-springboot.xml pattern: ? dateformat: MM-dd HH:mm:ss
1.4. 自定義日志攔截器:LogInterceptor.java
public class LogInterceptor implements HandlerInterceptor {
?
?private static final String TRACE_ID = "TRACE_ID";
?
?@Override
?public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
? ?// 使用UUID自動(dòng)生成鏈路ID
? ?String tid = UUID.randomUUID().toString().replace("-", "");
? ?// 客戶端可以傳入鏈路ID,需要唯一性
? ?String traceId = request.getHeader(TRACE_ID);
? ?if (!StringUtils.isEmpty(traceId)) {
? ? ?tid = request.getHeader(TRACE_ID);
? }
? ?// MDC(Mapped Diagnostic Context)診斷上下文映射,是@Slf4j提供的一個(gè)支持動(dòng)態(tài)打印日志信息的工具
? ?MDC.put(TRACE_ID, tid);
? ?return true;
}
?
?@Override
?public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
? ?MDC.remove(TRACE_ID);
}
}自定義的攔截器需要實(shí)現(xiàn)HandlerInterceptor.java接口,然后重寫preHandle方法;MDC(Mapped Diagnostic Context)診斷上下文映射,@Slf4j提供的動(dòng)態(tài)打印日志工具。
1.5. 添加攔截器:WebConfigurerAdapter.java
@Configuration
public class WebConfigurerAdapter implements WebMvcConfigurer {
?
?@Bean
?public LogInterceptor logInterceptor() {
? ?return new LogInterceptor();
}
?
?@Override
?public void addInterceptors(InterceptorRegistry registry) {
? ?registry.addInterceptor(logInterceptor())
? ? ?// 自定義需要攔截的和不需要攔截的
? ? .addPathPatterns("/**")
? ? .excludePathPatterns("/test***.html");
?
}
}經(jīng)過上述的幾個(gè)步驟,基本上就可以簡單的將同一次的業(yè)務(wù)調(diào)用鏈上的日志串聯(lián)起來了。
2. 測(cè)試驗(yàn)證
簡單的寫一個(gè)測(cè)試類:TestController.java
@RestController
@Slf4j
public class TestController {
?
?@Resource(name = "userService")
?private IUserService userService;
?
?@PostMapping("/test")
?public String testTrace01(@RequestParam("name") final String name) {
? ?log.info("入?yún)?name={}", name);
? ?testTrace02();
? ?log.info("調(diào)用結(jié)束name={}", name);
? ?return "Hello," + name;
}
}使用Postman調(diào)用接口:http://localhost:8080/test?name=張三
控制臺(tái)的輸出如下:

至此,一個(gè)最簡單的日志串聯(lián)就做好了
到此這篇關(guān)于SpringBoot實(shí)現(xiàn)簡單的日志鏈路追蹤的文章就介紹到這了,更多相關(guān)SpringBoot日志鏈路追蹤內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
使用Java實(shí)現(xiàn)通用樹形結(jié)構(gòu)構(gòu)建工具類
這篇文章主要為大家詳細(xì)介紹了如何使用Java實(shí)現(xiàn)通用樹形結(jié)構(gòu)構(gòu)建工具類,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2025-03-03
基于獲取JAVA路徑,包括CLASSPATH外的路徑的方法詳解
本篇文章是對(duì)獲取JAVA路徑,包括CLASSPATH外的路徑的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
解決Spring Cloud中Feign/Ribbon第一次請(qǐng)求失敗的方法
這篇文章主要給大家介紹了關(guān)于解決Spring Cloud中Feign/Ribbon第一次請(qǐng)求失敗的方法,文中給出了三種解決的方法,大家可以根據(jù)需要選擇對(duì)應(yīng)的方法,需要的朋友們下面來一起看看吧。2017-02-02
一招教你使用Java執(zhí)行g(shù)roovy腳本的兩種方式
本文主要介紹了一招教你使用Java執(zhí)行g(shù)roovy腳本的兩種方式,一種是通過腳本引擎ScriptEngine提供的eval(String)方法執(zhí)行腳本內(nèi)容,一種是執(zhí)行g(shù)roovy腳本,感興趣的可以了解一下2023-09-09
Springcloud中的Nacos?Config服務(wù)配置流程分析
這篇文章主要介紹了Springcloud中的Nacos?Config服務(wù)配置,本文以用戶微服務(wù)為例,進(jìn)行統(tǒng)一的配置,結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2022-09-09
SpringSecurity實(shí)現(xiàn)訪問控制url匹配
本文主要介紹了SpringSecurity實(shí)現(xiàn)訪問控制url匹配,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08

