全局請(qǐng)求添加TraceId輕松看日志
引言
不知道大家有沒(méi)有一堆日志就是定位不到那塊是異常部分,接口錯(cuò)誤無(wú)法復(fù)現(xiàn),也找不到報(bào)錯(cuò)信息等比較棘手的問(wèn)題。
其實(shí)解決上面的問(wèn)題很簡(jiǎn)單,只要我們?yōu)槊恳粋€(gè)請(qǐng)求都分配一個(gè)唯一的 RequestId 或者叫 TraceId ,一旦出了問(wèn)題,只需要拿著 Id 去日志里一搜,妖魔鬼怪立馬原形畢露。
對(duì)于分布式鏈路追蹤,有很多開(kāi)源中間件,本文主要通過(guò) logback 的 MDC 實(shí)現(xiàn)。
請(qǐng)求攔截器
@Component
public class TraceIdInterceptor extends HandlerInterceptorAdapter {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
String traceId = String.format("%s - %s",request.getRequestURI(), UUID.fastUUID().toString(true));
MDC.put("traceId", traceId);
return true;
}
注冊(cè)攔截器
@Component
@RequiredArgsConstructor
public class GlobalWebMvcConfigurer implements WebMvcConfigurer {
private final TraceIdInterceptor traceIdInterceptor;
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(traceIdInterceptor);
}
}
統(tǒng)一返回值
@Data
@NoArgsConstructor
@AllArgsConstructor
public class CommonResponse<T> implements Serializable {
/** 錯(cuò)誤碼 */
private Integer code;
/** 錯(cuò)誤消息 */
private String message;
/** 泛型響應(yīng)數(shù)據(jù) */
private T Data;
private String traceId;
public CommonResponse(Integer code, String message) {
this.code = code;
this.message = message;
this.traceId = MDC.get("traceId");
}
}
日志配置
logback.xml
<?xml version="1.0" encoding="UTF-8"?>
<configuration>
<property name="LOG_PATTERN"
value="%d{yyyy-MM-dd} %d{HH:mm:ss.SSS} [%highlight(%-5level)] [%boldYellow(%X{traceId})] [%boldYellow(%thread)] %boldGreen(%logger{36} %F.%L) %msg%n">
</property>
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
<encoder>
<pattern>${LOG_PATTERN}</pattern>
</encoder>
<!-- 控制臺(tái)打印INFO及以上級(jí)別的日志 -->
<filter class="ch.qos.logback.classic.filter.ThresholdFilter">
<level>INFO</level>
</filter>
</appender>
<root>
<appender-ref ref="STDOUT"/>
</root>
</configuration>
測(cè)試
接口返回值
{
"code": 0,
"message": "",
"traceId": "/commerce-user/user/findById - 73e470120ef24d57a111de6b671d030b",
"data": {
"id": 1,
"username": "test1",
"password": "test",
"extraInfo": "{}",
"createTime": "2022-06-20T09:23:18.000+00:00",
"updateTime": "2022-06-20T09:23:18.000+00:00",
"balance": 0
}
}
日志

如此,如果線上有接口出現(xiàn)問(wèn)題,拿著 traceId 到日志文件搜索,就能檢索到該請(qǐng)求的日志調(diào)用鏈路,處理問(wèn)題就變得簡(jiǎn)單了。
異步調(diào)用配置
因?yàn)槭?ThreadLocal ,異步任務(wù)必然是獲取不到 traceId 的,需要再線程池配置中手動(dòng)添加。

線程池配置
@Configuration
public class AsyncConfig implements AsyncConfigurer {
@Bean
@Override
public Executor getAsyncExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setCorePoolSize(10);
executor.setMaxPoolSize(20);
executor.setQueueCapacity(20);
executor.setKeepAliveSeconds(60);
executor.setThreadNamePrefix("user-async-");
// 等待所有任務(wù)結(jié)果候再關(guān)閉線程池
executor.setWaitForTasksToCompleteOnShutdown(true);
executor.setAwaitTerminationSeconds(60);
// 定義拒絕策略
executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
//設(shè)置線程裝飾器
executor.setTaskDecorator(runnable -> ThreadMdcUtils.wrapAsync(runnable, MDC.getCopyOfContextMap()));
// 初始化線程池, 初始化 core 線程
executor.initialize();
return executor;
}
}
public class ThreadMdcUtils {
public static Runnable wrapAsync(Runnable task, Map<String,String> context){
return () -> {
if(context==null){
MDC.clear();
}else {
MDC.setContextMap(context);
}
if(MDC.get("traceId")==null){
MDC.put("traceId", UUID.fastUUID().toString(true));
}
try {
task.run();
}finally {
MDC.clear();
}
};
}
}
再次測(cè)試

以上就是全局請(qǐng)求添加TraceId輕松看日志的詳細(xì)內(nèi)容,更多關(guān)于全局請(qǐng)求添加TraceId的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java登錄功能實(shí)現(xiàn)token生成與驗(yàn)證
這篇文章介紹了Java登錄功能實(shí)現(xiàn)token生成與驗(yàn)證,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-12-12
解決idea中svn提交時(shí)performing vcs refresh時(shí)間很長(zhǎng)的問(wèn)題
這篇文章主要介紹了解決idea中svn提交時(shí)performing vcs refresh時(shí)間很長(zhǎng)的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-09-09
Java如何基于IO流實(shí)現(xiàn)同一文件讀寫(xiě)操作
這篇文章主要介紹了Java如何基于IO流實(shí)現(xiàn)文件讀寫(xiě)操作,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-10-10
解決Spring Mvc中對(duì)象綁定參數(shù)重名的問(wèn)題
最近在工作中遇到了參數(shù)綁定的一個(gè)問(wèn)題,發(fā)現(xiàn)網(wǎng)上這方面的資料較少,索性自己來(lái)總結(jié)下,下面這篇文章主要給大家介紹了關(guān)于如何解決Spring Mvc中對(duì)象綁定參數(shù)重名問(wèn)題的相關(guān)資料,需要的朋友可以參考借鑒,下面來(lái)一起看看吧。2017-08-08
Mybatis調(diào)用MySQL存儲(chǔ)過(guò)程的簡(jiǎn)單實(shí)現(xiàn)
本篇文章主要介紹了Mybatis調(diào)用MySQL存儲(chǔ)過(guò)程的簡(jiǎn)單實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。2017-04-04
Spring中PathMatcher路徑匹配器的實(shí)現(xiàn)
Spring框架中的PathMatcher是一個(gè)接口,本文主要介紹了Spring中PathMatcher路徑匹配器的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2024-07-07
Spring為何要用三級(jí)緩存來(lái)解決循環(huán)依賴問(wèn)題
這篇文章主要給大家介紹了關(guān)于Spring為何要用三級(jí)緩存來(lái)解決循環(huán)依賴問(wèn)題的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10

