SpringBoot實(shí)現(xiàn)簡(jiǎ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è)簡(jiǎn)單的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è)步驟,基本上就可以簡(jiǎn)單的將同一次的業(yè)務(wù)調(diào)用鏈上的日志串聯(lián)起來了。
2. 測(cè)試驗(yàn)證
簡(jiǎ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è)最簡(jiǎn)單的日志串聯(lián)就做好了
到此這篇關(guān)于SpringBoot實(shí)現(xiàn)簡(jiǎn)單的日志鏈路追蹤的文章就介紹到這了,更多相關(guān)SpringBoot日志鏈路追蹤內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Servlet簡(jiǎn)單實(shí)現(xiàn)登錄功能
這篇文章主要為大家詳細(xì)介紹了Servlet簡(jiǎn)單實(shí)現(xiàn)登錄功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-03-03使用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-09Springcloud中的Nacos?Config服務(wù)配置流程分析
這篇文章主要介紹了Springcloud中的Nacos?Config服務(wù)配置,本文以用戶微服務(wù)為例,進(jìn)行統(tǒng)一的配置,結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2022-09-09SpringSecurity實(shí)現(xiàn)訪問控制url匹配
本文主要介紹了SpringSecurity實(shí)現(xiàn)訪問控制url匹配,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08