Spring如何實現(xiàn)輸出帶動態(tài)標簽的日志
版權說明: 本文由CSDN博主keep丶原創(chuàng),轉載請保留此塊內容在文首。
原文地址: https://blog.csdn.net/qq_38688267/article/details/145022997
背景
部分業(yè)務代碼會被多個模塊調用,此時該部分代碼輸出的日志無法直觀看出是從哪個模塊調用的,因此提出動態(tài)標簽日志需求,效果如下:

底層原理
業(yè)務代碼起始時通過ThreadLocal存儲當前業(yè)務標簽值,后續(xù)日志輸出時,插入緩存的業(yè)務標簽到輸出的日志中。即可實現(xiàn)該需求。
實現(xiàn)方案
Tag緩存實現(xiàn)
private static final ThreadLocal<String> logTagCache = new ThreadLocal<>();
/**
* 獲取緩存的標簽值
* <b style="color:red">只有添加了@BizLog注解的方法內才可用</b>
*/
static String getCacheTag() {
String temp = logTagCache.get();
if (temp == null) {
log.warn("[LogHelper] 緩存標簽為空, 請及時配置@BizLog注解或手動緩存標簽.");
return DEFAULT_TAG;
}
return temp;
}
static void cacheTag(String logTag) {
logTagCache.set(logTag);
}
/**
* 清空當前線程緩存
* <br/>
* <b>使用set()或init()之后,請在合適的地方調用clean(),一般用try-finally語法在finally塊中調用</b>
*/
static void cleanCache() {
logTagCache.remove();
}
封裝注解通過AOP實現(xiàn)日志緩存
注解定義
/**
* 啟動業(yè)務日志注解
*
* @author zeng.zf
*/
@Target({ElementType.METHOD, ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
public @interface BizLog {
/**
* 日志標簽值
* <p>
* 如果不給值則默認輸出類型方法名,給值用給的值<br/>
* <b>會緩存標簽值,可使用{@code LogHelper.xxxWCT()}方法</b>
* @see cn.xxx.log.util.LogHelper.WCT#catchLog(Logger, Runnable)
* @see cn.xxx.log.util.LogHelper.WCT#log(String, Object...)
*/
String value();
}AOP切面配置
/**
* {@link BizLog} 切面,記錄業(yè)務鏈路
*
* @author zzf
*/
@Aspect
@Order// 日志的AOP邏輯放最后執(zhí)行
public class BizLogAspect {
@Around("@annotation(bizLog)")
public Object around(ProceedingJoinPoint joinPoint, BizLog bizLog) throws Throwable {
try {
LogHelper.UTIL.cacheTag(bizLog.value());
return joinPoint.proceed();
} finally {
LogHelper.UTIL.cleanCache();
}
}
封裝行為參數(shù)通用方法實現(xiàn)
/**
* 緩存給定tag后執(zhí)行給定方法<br/>
* 使用:{@code LogHelper.beginTrace("AddUser", () -> userService.addUser(user))}
*
* @param tag 日志標簽
* @param runnable 執(zhí)行內容
*/
static void beginTrace(String tag, Runnable runnable) {
try {
cacheTag(tag);
runnable.run();
} finally {
cleanCache();
}
}
/**
* 緩存給定tag后執(zhí)行給定方法<br/>
* 使用:{@code return LogHelper.beginTrace("AddUser", () -> userService.addUser(user))}
*
* @param tag 日志標簽
* @param supplier 有返回值的執(zhí)行內容
*/
static <T> T beginTrace(String tag, Supplier<T> supplier) {
try {
cacheTag(tag);
return supplier.get();
} finally {
cleanCache();
}
}手動緩存Tag值
- 非Bean方法可通過手動調用LogHelper.UTIL.beginTrace()方法實現(xiàn)@BizLog相同功能。
- 也可以參考方法手寫cacheTag()和cleanCache()實現(xiàn)該功能。
- 一般不建議這么做,使用工具類方法最好。
- 如果runnable參數(shù)會拋異常的情況下就不適合用工具方法,此時可以手寫。
- 手寫時必須使用try-finally塊,并在finally塊中調用cleanCache()。

整理代碼,封裝通用LogHelper類
/**
* 日志輸出輔助類
* <br/>
* 注意:所有格式化參數(shù)在格式化時都是調用其toString()方法<br/>
* 因此對象需要重寫toString()方法或者使用{@code JSONUtil.toJsonStr()}轉成JSON字符串。<br/>
* <br/>
* <b>如果自行輸出日志,請按照該格式: {@code "[TAG][SUB_TAG] CONTENT"}</b>
* <p>如: 1. {@code "[AddUser] add success"}</p>
* <p>  2. {@code "[AddUser][GenRole] add success"}</p>
* <p>  2. {@code "[AddUser][BizException] 用戶名重復"}</p>
* <p>更多請參考源文件中的LogHelperTest測試類</p>
*/
@Slf4j
public class LogHelper {
/**
* 緩存{@link cn.xxx.log.core.aop.log.BizLog} 注解的value值
*/
private static final ThreadLocal<String> logTagCache = new ThreadLocal<>();
private static final String DEFAULT_TAG = "TAG_NOT_CONFIG";
/*===========以下為工具方法,提供Tag緩存相關方法============*/
public interface UTIL {
/**
* 緩存給定tag后執(zhí)行給定方法<br/>
* 使用:{@code LogHelper.beginTrace("AddUser", () -> userService.addUser(user))}
*
* @param tag 日志標簽
* @param runnable 執(zhí)行內容
*/
static void beginTrace(String tag, Runnable runnable) {
try {
cacheTag(tag);
runnable.run();
} finally {
cleanCache();
}
}
/**
* 緩存給定tag后執(zhí)行給定方法<br/>
* 使用:{@code return LogHelper.beginTrace("AddUser", () -> userService.addUser(user))}
*
* @param tag 日志標簽
* @param supplier 有返回值的執(zhí)行內容
*/
static <T> T beginTrace(String tag, Supplier<T> supplier) {
try {
cacheTag(tag);
return supplier.get();
} finally {
cleanCache();
}
}
/**
* 緩存給定tag后執(zhí)行給定方法,提供默認異常處理<br/>
* 使用:{@code LogHelper.catchBeginTrace(log, "AddUser", () -> userService.addUser(user))}
*
* @param tag 日志標簽
* @param runnable 執(zhí)行內容
*/
static void catchBeginTrace(Logger logger, String tag, Runnable runnable) {
try {
cacheTag(tag);
WCT.catchLog(logger, runnable);
} finally {
cleanCache();
}
}
/**
* 緩存給定tag后執(zhí)行給定方法,提供默認異常處理<br/>
* 使用:{@code return LogHelper.catchBeginTrace(log, "AddUser", () -> userService.addUser(user))}
*
* @param tag 日志標簽
* @param supplier 有返回值的執(zhí)行內容
*/
static <T> @Nullable T catchBeginTrace(Logger logger, String tag, Supplier<T> supplier) {
try {
cacheTag(tag);
return WCT.catchLog(logger, supplier);
} finally {
cleanCache();
}
}
/**
* 獲取緩存的標簽值
* <b style="color:red">只有添加了@BizLog注解的方法內才可用</b>
*/
static String getCacheTag() {
String temp = logTagCache.get();
if (temp == null) {
log.warn("[LogHelper] 緩存標簽為空, 請及時配置@BizLog注解或手動緩存標簽.");
return DEFAULT_TAG;
}
return temp;
}
static void cacheTag(String logTag) {
logTagCache.set(logTag);
}
/**
* 清空當前線程緩存
* <br/>
* <b>使用set()或init()之后,請在合適的地方調用clean(),一般用try-finally語法在finally塊中調用</b>
*/
static void cleanCache() {
logTagCache.remove();
}
}
/*=========================以下為基礎方法,提供基礎日志輸出方法=======================*/
public interface BASIC {
/**
* 標簽日志<br/>
* 例:
* {@code LogHelper.tag("AddUser", "GenRole", "add success, user id = {}, name = {}", 1L, "zs")}<br/>
* 返回 {@code "[AddUser][GenRole] add success, user id = 1, name = zs"}
*
* @param tag 標簽
* @param content 需要格式化內容
* @param arg 格式化參數(shù)
* @return 最終日志
*/
static String tag(String tag, String content, Object... arg) {
return StrUtil.format("[{}] {}", tag, StrUtil.format(content, arg));
}
/**
* 兩級標簽日志<br/>
* 例:
* {@code LogHelper.tag("AddUser", "GenRole", "add success")}<br/>
* 返回 {@code "[AddUser][GenRole] add success"}
*
* @param tag 標簽
* @param subTag 子標簽
* @param content 內容
* @return 最終日志
*/
static String doubleTag(String tag, String subTag, String content, Object... args) {
return StrUtil.format("[{}][{}] {}", tag, subTag, StrUtil.format(content, args));
}
/**
* 業(yè)務異常tag日志內容生成
*/
static String bizExTag(String tag, BizExceptionMark bizException) {
return StrUtil.format("[{}][{}] code={},msg={}", tag, bizException.getClass().getSimpleName(),
bizException.getCode(), bizException.getMsg());
}
/**
* 業(yè)務異常tag日志內容生成
*/
static String bizExTag(String tag, BizExceptionMark bizException, String extraInfo, Object... args) {
return StrUtil.format("[{}][{}] code={},msg={}, extraInfo={{}}", tag,
bizException.getClass().getSimpleName(), bizException.getCode(),
bizException.getMsg(), StrUtil.format(extraInfo, args));
}
/**
* 業(yè)務異常tag日志內容生成
*/
static String bizEx(BizExceptionMark bizException) {
return StrUtil.format("[{}] code={},msg={}", bizException.getClass().getSimpleName(),
bizException.getCode(), bizException.getMsg());
}
/**
* 運行時異常tag日志內容生成
*/
static String otherExTag(String tag, Exception e) {
return StrUtil.format("[{}][{}] msg={}, stackTrace={}", tag, e.getClass().getSimpleName(), e.getMessage(),
TraceUtils.getStackTraceStr(e.getStackTrace()));
}
/**
* 運行時異常tag日志內容生成
*/
static String otherExTag(String tag, Exception e, String extraInfo, Object... args) {
return StrUtil.format("[{}][{}] msg={}, extraInfo={{}}, stackTrace={}", tag, e.getClass().getSimpleName(),
e.getMessage(), StrUtil.format(extraInfo, args),
TraceUtils.getStackTraceStr(e.getStackTrace()));
}
/**
* 其他異常tag日志內容生成
*/
static String otherEx(Exception e) {
return StrUtil.format("[{}] msg={}, stackTrace={}", e.getClass().getSimpleName(), e.getMessage(),
TraceUtils.getStackTraceStr(e.getStackTrace()));
}
/**
* 通用標簽日志包裝<br/>
* <p>使用案例:</p>
* 1. {@code tagLogWrap(() -> bizMethod(param1, param2))}<br/>
* 2.
* <pre><code>
* \@Slf4j
* public class UserServiceImpl{
*
* public void addUsers(List<User> users) {
* for(user in users) {
* LogHelper.tagLogWrap(log, "AddUsers", () -> addUser(user));
* }
* }
* public void addUser(User user) {
* xxx
* xxx
* ...
* }
* }
* </code></pre>
*
* @param tag 日志標簽
* @param runnable 你要執(zhí)行的無返回值方法
*/
static void catchLog(Logger logger, String tag, Runnable runnable) {
try {
runnable.run();
} catch (Exception e) {
if (e instanceof BizExceptionMark) {
logger.warn(bizExTag(tag, (BizExceptionMark) e));
} else {
logger.error(otherExTag(tag, e));
}
}
}
/**
* 通用標簽日志包裝<br/>
* <p>使用案例:</p>
* 1. {@code tagLogWrap(() -> bizMethod(param1, param2))}<br/>
* 2.
* <pre><code>
* \@Slf4j
* public class UserServiceImpl{
*
* public void addUsers(List<User> users) {
* for(user in users) {
* LogHelper.tagLogWrap(
* log,
* "AddUsers",
* () -> addUser(user),
* "id = {}, name={}",
* user.getId(),
* user.getName()
* );
* }
* }
* public void addUser(User user) {
* xxx
* xxx
* ...
* }
* }
* </code></pre>
*
* @param tag 日志標簽
* @param runnable 你要執(zhí)行的無返回值方法
*/
static void catchLog(Logger logger, String tag, Runnable runnable, String extraInfo, Object... args) {
try {
runnable.run();
} catch (Exception e) {
if (e instanceof BizExceptionMark) {
logger.warn(bizExTag(tag, (BizExceptionMark) e, extraInfo, args));
} else {
logger.error(otherExTag(tag, e, extraInfo, args));
}
}
}
/**
* 通用標簽日志包裝<br/>
* <p>使用案例:</p>
* 1. {@code return tagLogWrap(() -> bizMethod(param1, param2))}<br/>
* 2.
* <pre><code>
* \@Slf4j
* public class UserServiceImpl{
*
* public List<User> getUserByIds(List<Long> ids) {
* return ids.map(id ->
* LogHelper.tagLogWrap(log, "getUserByIds", () -> getUserById(id))
* ).collect(Collectors.toList());
* }
* public User getUserById(Long userId) {
* xxx
* xxx
* ...
* return user;
* }
* }
* </code></pre>
*
* @param tag 日志標簽
* @param supplier 你要執(zhí)行的有返回值方法
* @return 方法執(zhí)行結果(可能為null)
*/
static <R> @Nullable R catchLog(Logger logger, String tag, Supplier<R> supplier) {
try {
return supplier.get();
} catch (Exception e) {
if (e instanceof BizExceptionMark) {
logger.warn(bizExTag(tag, (BizExceptionMark) e));
} else {
logger.error(otherExTag(tag, e));
}
}
return null;
}
/**
* 通用標簽日志包裝<br/>
* <p>使用案例:</p>
* 1. {@code return tagLogWrap(() -> bizMethod(param1, param2))}<br/>
* 2.
* <pre><code>
* \@Slf4j
* public class UserServiceImpl{
*
* public List<User> getUserByIds(List<Long> ids) {
* return ids.map(id ->
* LogHelper.tagLogWrap(
* log,
* "getUserByIds",
* () -> getUserById(id),
* "id={}",
* id
* )
* ).collect(Collectors.toList());
* }
* public User getUserById(Long userId) {
* xxx
* xxx
* ...
* return user;
* }
* }
* </code></pre>
*
* @param tag 日志標簽
* @param supplier 你要執(zhí)行的有返回值方法
* @return 方法執(zhí)行結果(可能為null)
*/
static <R> @Nullable R catchLog(Logger logger, String tag, Supplier<R> supplier, String extraInfo,
Object... args) {
try {
return supplier.get();
} catch (Exception e) {
if (e instanceof BizExceptionMark) {
logger.warn(bizExTag(tag, (BizExceptionMark) e, extraInfo, args));
} else {
logger.error(otherExTag(tag, e, extraInfo, args));
}
}
return null;
}
}
/*===================以下為基于緩存標簽的方法,理論上上方基礎方法都要在下面有對應的方法==================*/
/* WCT = with cached tag */
public interface WCT {
/**
* <b style="color:red">只有添加了@BizLog注解的方法內才可用</b>
*/
static String log(String content, Object... args) {
return BASIC.tag(getCacheTag(), content, args);
}
/**
* <b style="color:red">只有添加了@BizLog注解的方法內才可用</b>
*/
static String sub(String subTag, String content, Object... args) {
return BASIC.doubleTag(getCacheTag(), subTag, content, args);
}
/**
* 業(yè)務異常tag日志內容生成
* <b style="color:red">只有添加了@BizLog注解的方法內才可用</b>
*/
static String bizEx(BizExceptionMark bizException) {
return BASIC.bizExTag(getCacheTag(), bizException);
}
/**
* 業(yè)務異常tag日志內容生成
* <b style="color:red">只有添加了@BizLog注解的方法內才可用</b>
*/
static String bizEx(BizExceptionMark bizException, String extraInfo, Object... args) {
return BASIC.bizExTag(getCacheTag(), bizException, extraInfo, args);
}
/**
* 運行時異常tag日志內容生成
* <b style="color:red">只有添加了@BizLog注解的方法內才可用</b>
*/
static String otherEx(Exception e) {
return BASIC.otherExTag(getCacheTag(), e);
}
/**
* 運行時異常tag日志內容生成
* <b style="color:red">只有添加了@BizLog注解的方法內才可用</b>
*/
static String otherEx(String tag, Exception e, String extraInfo, Object... args) {
return BASIC.otherExTag(tag, e, extraInfo, args);
}
/**
* 通用標簽日志包裝<br/>
* <b style="color:red">只有添加了@BizLog注解的方法內才可用</b>
*
* @param runnable 你要執(zhí)行的無返回值方法
*/
static void catchLog(Logger logger, Runnable runnable) {
BASIC.catchLog(logger, getCacheTag(), runnable);
}
/**
* 通用標簽日志包裝<br/>
* <b style="color:red">只有添加了@BizLog注解的方法內才可用</b>
*
* @param runnable 你要執(zhí)行的無返回值方法
*/
static void catchLog(Logger logger, Runnable runnable, String extraInfo, Object... args) {
BASIC.catchLog(logger, getCacheTag(), runnable, extraInfo, args);
}
/**
* 通用標簽日志包裝<br/>
* <b style="color:red">只有添加了@BizLog注解的方法內才可用</b>
*
* @param supplier 你要執(zhí)行的有返回值方法
* @return 方法執(zhí)行結果(可能為null)
*/
static <R> @Nullable R catchLog(Logger logger, Supplier<R> supplier) {
return BASIC.catchLog(logger, getCacheTag(), supplier);
}
/**
* 通用標簽日志包裝<br/>
* <b style="color:red">只有添加了@BizLog注解的方法內才可用</b>
*
* @param supplier 你要執(zhí)行的有返回值方法
* @return 方法執(zhí)行結果(可能為null)
*/
static <R> @Nullable R catchLog(Logger logger, Supplier<R> supplier, String extraInfo, Object... args) {
return BASIC.catchLog(logger, getCacheTag(), supplier, extraInfo, args);
}
}
}相關資料
Spring實現(xiàn)Logback日志模板設置動態(tài)參數(shù)
Spring實現(xiàn)通過工具類統(tǒng)一輸出日志(不改變日志類信息)
到此這篇關于Spring實現(xiàn)輸出帶動態(tài)標簽的日志的文章就介紹到這了,更多相關Spring動態(tài)標簽的日志內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
淺談@RequestParam(required = true)的誤區(qū)
這篇文章主要介紹了@RequestParam(required = true)的誤區(qū),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-11-11
記錄jdk21連接SQLServer因為TLS協(xié)議報錯問題
在使用Druid連接池連接SQL Server時,可能會遇到因TLS版本不匹配導致的連接失敗問題,具體表現(xiàn)為客戶端使用TLS1.3或TLS1.2,而SQL Server僅支持TLS1.0,導致無法建立安全連接,解決方法是修改JDK的安全配置,啟用TLS1.02024-10-10
Java編程rabbitMQ實現(xiàn)消息的收發(fā)
RabbitMQ是一個在AMQP基礎上完成的,可復用的企業(yè)消息系統(tǒng),本文通過實例來給大家分享通過操作rabbitMQ實現(xiàn)消息的收發(fā),感興趣的朋友可以參考下。2017-09-09
Java模擬實現(xiàn)HTTP服務器項目實戰(zhàn)
本文主要介紹了Java模擬實現(xiàn)HTTP服務器項目實戰(zhàn),文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-03-03
PowerJob的TimingStrategyHandler工作流程源碼解讀
這篇文章主要為大家介紹了PowerJob的TimingStrategyHandler工作流程源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2024-01-01

