Spring实现通过工具类统一输出日志(不改变日志类信息)

devtools/2025/1/13 2:50:46/

版权说明: 本文由CSDN博主keep丶原创,转载请保留此块内容在文首。
原文地址: https://blog.csdn.net/qq_38688267/article/details/145022997

背景

  实现输出带动态标签的日志需求后,实际操作过程中,输出日志的代码为:log.info(LogHelper.WTC.tag("log content")),用起来不太方便,提出直接通过LogHelper输出日志,但是常规实现通过LogHelper输出日志会导致日志中的类信息是LogHelper的,解决该问题后可实现LogHelper.WTC.logTag("log content")的方式实现日志输出。预期效果如下:
在这里插入图片描述

实现原理

重写日志类信息输出转换类

  重写Logback类信息输出转换类,通过回溯堆栈的方式找到调用LogHelper方法的堆栈,输出调用方的类信息。

java">import ch.qos.logback.classic.spi.ILoggingEvent;
import ch.qos.logback.core.pattern.DynamicConverter;
import cn.xx.log.util.LogHelper;/*** Logback日志 类信息动态转换类* <br/>* 在{@code logback-spring.xml}中使用:<br/>* {@code <define name="traceId" class="cn.xx.log.core.log.TraceLogConverter"/>}* @author zeng.zf*/
public class ClassMethodLineConverter extends DynamicConverter<ILoggingEvent> {@Overridepublic String convert(ILoggingEvent event) {StackTraceElement[] stackTrace = event.getCallerData();if (stackTrace != null && stackTrace.length > 0) {int idx = LogHelper.UTIL.stackDepth();if(idx > stackTrace.length - 1) {idx = 0;}StackTraceElement element = stackTrace[idx];return String.format("%s.%s(%s:%d)",element.getClassName(),element.getMethodName(),element.getFileName(),element.getLineNumber());}return "unknown";}
}

缓存传递堆栈深度

在调用LogHelper输出日志时,缓存当前堆栈深度:
在这里插入图片描述
在这里插入图片描述

springxml_50">配置logback-spring.xml

<configuration><!-- 引用 Spring Boot 的 logback 基础配置 --><include resource="org/springframework/boot/logging/logback/defaults.xml" /><!-- 引用 traceId 动态参数转换类 --><define name="traceId" class="cn.xx.log.core.log.TraceLogConverter"/><!-- 注册自定义转换器 --><conversionRule conversionWord="CDC" converterClass="cn.xx.log.core.log.ClassMethodLineConverter" /><!-- 格式化输出:%d 表示日期,%X{tid} SkWalking 链路追踪编号,%thread 表示线程名,%-5level:级别从左显示 5 个字符宽度,%msg:日志消息,%n是换行符 --><property name="PATTERN_CONSOLE" value="%green(%d{${yyyy-MM-dd HH:mm:ss.SSS}})|%highlight(%-5level)|%t|%magenta(%X{traceId})|%cyan(%CDC): %m%n"/><property name="PATTERN_FILE" value="%d{${yyyy-MM-dd HH:mm:ss.SSS}}|${LOG_LEVEL_PATTERN:-%4p}|%t|%X{traceId}|%CDC: %m%n"/><property name="LOG_FILE" value="trace.log"/>

至此,实现预期功能。

整理Loghelper类代码

java">package cn.xx.log.util;import cn.xx.log.core.exception.BizExceptionMark;
import cn.hutool.core.util.StrUtil;
import lombok.extern.slf4j.Slf4j;
import org.slf4j.Logger;
import org.springframework.lang.Nullable;import java.util.function.Consumer;
import java.util.function.Supplier;import static cn.xx.log.util.LogHelper.UTIL.getCacheTag;/*** 日志输出辅助类* <br/>* 注意:所有格式化参数在格式化时都是调用其toString()方法<br/>* 因此对象需要重写toString()方法或者使用{@code JSONUtil.toJsonStr()}转成JSON字符串。<br/>* <br/>* <b>如果自行输出日志,请按照该格式: {@code "[TAG][SUB_TAG] CONTENT"}</b>* <p>如: 1. {@code "[AddUser] add success"}</p>* <p>&emsp;&emsp;2. {@code "[AddUser][GenRole] add success"}</p>* <p>&emsp;&emsp;2. {@code "[AddUser][BizException] 用户名重复"}</p>* <p>更多请参考源文件中的LogHelperTest测试类</p>*/
@Slf4j
public class LogHelper {/*** 缓存{@link cn.xx.log.core.aop.log.BizLog} 注解的value值*/private static final ThreadLocal<String> logTagCache = new ThreadLocal<>();private static final ThreadLocal<Integer> stackDepth = new ThreadLocal<>();private static final String DEFAULT_TAG = "TAG_NOT_CONFIG";private static void helperLog(Consumer<String> logger, String logContent) {stackDepth.set(4);logger.accept(logContent);stackDepth.remove();}private static void helperLog(Runnable runnable) {stackDepth.set(4);runnable.run();stackDepth.remove();}private static void helperLog(Runnable runnable, int depth) {stackDepth.set(4 + depth);runnable.run();stackDepth.remove();}/*===========以下为工具方法,提供Tag缓存相关方法============*/public interface UTIL {/*** 缓存给定tag后执行给定方法<br/>* 使用:{@code LogHelper.beginTrace("AddUser", () -> userService.addUser(user))}** @param tag      日志标签* @param runnable 执行内容*/static void beginTrace(String tag, Runnable runnable) {try {cacheTag(tag);runnable.run();} finally {cleanCache();}}/*** 缓存给定tag后执行给定方法<br/>* 使用:{@code return LogHelper.beginTrace("AddUser", () -> userService.addUser(user))}** @param tag      日志标签* @param supplier 有返回值的执行内容*/static <T> T beginTrace(String tag, Supplier<T> supplier) {try {cacheTag(tag);return supplier.get();} finally {cleanCache();}}/*** 缓存给定tag后执行给定方法,提供默认异常处理<br/>* 使用:{@code LogHelper.catchBeginTrace(log, "AddUser", () -> userService.addUser(user))}** @param tag      日志标签* @param runnable 执行内容*/static void catchBeginTrace(String tag, Runnable runnable) {try {cacheTag(tag);runnable.run();} catch (Exception e) {if (e instanceof BizExceptionMark) {helperLog(() -> log.warn(WCT.bizEx((BizExceptionMark) e)));} else {helperLog(() -> log.error(WCT.otherEx(e)));}} finally {cleanCache();}}/*** 缓存给定tag后执行给定方法,提供默认异常处理<br/>* 使用:{@code return LogHelper.catchBeginTrace(log, "AddUser", () -> userService.addUser(user))}** @param tag      日志标签* @param supplier 有返回值的执行内容*/static <T> @Nullable T catchBeginTrace(String tag, Supplier<T> supplier) {try {cacheTag(tag);return supplier.get();} catch (Exception e) {if (e instanceof BizExceptionMark) {helperLog(() -> log.warn(WCT.bizEx((BizExceptionMark) e)));} else {helperLog(() -> log.error(WCT.otherEx(e)));}} finally {cleanCache();}return null;}/*** 获取缓存的标签值* <b style="color:red">只有缓存了标签之后才可用</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);}static void cacheTagIfAbsent(String logTag) {if (logTagCache.get() == null) {logTagCache.set(logTag);}}/*** 清空当前线程缓存* <br/>* <b>使用set()或init()之后,请在合适的地方调用clean(),一般用try-finally语法在finally块中调用</b>*/static void cleanCache() {logTagCache.remove();}static int stackDepth() {return stackDepth.get() == null ? 0 : stackDepth.get();}}/*=========================以下为基础方法,提供基础日志输出方法=======================*/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     格式化参数* @return 最终日志*/static String tag(String tag, String content, Object... arg) {return StrUtil.format("[{}] {}", tag, StrUtil.format(content, arg));}static void logTag(String tag, String content, Object... arg) {helperLog(() -> log.info(tag(tag, 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));}static void logDoubleTag(String tag, String subTag, String content, Object... args) {helperLog(() -> log.info(doubleTag(tag, subTag, content, args)));}/*** 业务异常tag日志内容生成*/static String bizEx(BizExceptionMark bizException) {return StrUtil.format("[{}] code={},msg={}",bizException.getClass().getSimpleName(),bizException.getCode(),bizException.getMsg());}static void logBizEx(BizExceptionMark bizException) {helperLog(() -> log.warn(bizEx(bizException)));}/*** 业务异常tag日志内容生成*/static String bizEx(String tag, BizExceptionMark bizException) {return StrUtil.format("[{}][{}] code={},msg={}",tag,bizException.getClass().getSimpleName(),bizException.getCode(),bizException.getMsg());}static void logBizEx(String tag, BizExceptionMark bizException) {helperLog(() -> log.warn(bizEx(tag, bizException)));}/*** 业务异常tag日志内容生成*/static String bizEx(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));}static void logBizEx(String tag, BizExceptionMark bizException, String extraInfo, Object... args) {helperLog(() -> log.warn(bizEx(tag, bizException, extraInfo, args)));}/*** 其他异常tag日志内容生成*/static String otherEx(Exception e) {return StrUtil.format("[{}] msg={}, stackTrace={}",e.getClass().getSimpleName(),e.getMessage(),TraceUtils.getStackTraceStr(e.getStackTrace()));}static void logOtherEx(Exception e) {helperLog(() -> log.error(otherEx(e)));}/*** 运行时异常tag日志内容生成*/static String otherEx(String tag, Exception e) {return StrUtil.format("[{}][{}] msg={}, stackTrace={}",tag,e.getClass().getSimpleName(),e.getMessage(),TraceUtils.getStackTraceStr(e.getStackTrace()));}static void logOtherEx(String tag, Exception e) {helperLog(() -> log.error(otherEx(tag, e)));}/*** 运行时异常tag日志内容生成*/static String otherEx(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()));}static void logOtherEx(String tag, Exception e, String extraInfo, Object... args) {helperLog(() -> log.error(otherEx(tag, e, extraInfo, args)));}/*** 通用标签日志包装<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 你要执行的无返回值方法*/static void catchLog(String tag, Runnable runnable) {try {runnable.run();} catch (Exception e) {if (e instanceof BizExceptionMark) {helperLog(() -> log.warn(bizEx(tag, (BizExceptionMark) e)));} else {helperLog(() -> log.error(otherEx(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 你要执行的无返回值方法*/static void catchLog(String tag, Runnable runnable, String extraInfo, Object... args) {try {runnable.run();} catch (Exception e) {if (e instanceof BizExceptionMark) {helperLog(() -> log.warn(bizEx(tag, (BizExceptionMark) e, extraInfo, args)));} else {helperLog(() -> log.error(otherEx(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 你要执行的有返回值方法* @return 方法执行结果(可能为null)*/static <R> @Nullable R catchLog(String tag, Supplier<R> supplier) {try {return supplier.get();} catch (Exception e) {if (e instanceof BizExceptionMark) {helperLog(() -> log.warn(bizEx(tag, (BizExceptionMark) e)));} else {helperLog(() -> log.error(otherEx(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 你要执行的有返回值方法* @return 方法执行结果(可能为null)*/static <R> @Nullable R catchLog(String tag, Supplier<R> supplier, String extraInfo, Object... args) {try {return supplier.get();} catch (Exception e) {if (e instanceof BizExceptionMark) {helperLog(() -> log.warn(bizEx(tag, (BizExceptionMark) e, extraInfo, args)));} else {helperLog(() -> log.error(otherEx(tag, e, extraInfo, args)));}}return null;}}/*===================以下为基于缓存标签的方法,理论上上方基础方法都要在下面有对应的方法==================*//* WCT = with cached tag *//*** <b style="color:red">只有缓存了标签之后才可用</b>*/public interface WCT {/*** <b style="color:red">只有缓存了标签之后才可用</b>*/static String tag(String content, Object... args) {return BASIC.tag(getCacheTag(), content, args);}static void logTag(String content, Object... args) {helperLog(() -> log.info(tag(content, args)));}/*** <b style="color:red">只有缓存了标签之后才可用</b>*/static String doubleTag(String subTag, String content, Object... args) {return BASIC.doubleTag(getCacheTag(), subTag, content, args);}static void logDoubleTag(String subTag, String content, Object... args) {helperLog(() -> log.info(doubleTag(subTag, content, args)));}/*** 业务异常tag日志内容生成* <b style="color:red">只有缓存了标签之后才可用</b>*/static String bizEx(BizExceptionMark bizException) {return BASIC.bizEx(getCacheTag(), bizException);}static void logBizEx(BizExceptionMark bizException) {helperLog(() -> log.warn(bizEx(bizException)));}/*** 业务异常tag日志内容生成* <b style="color:red">只有缓存了标签之后才可用</b>*/static String bizEx(BizExceptionMark bizException, String extraInfo, Object... args) {return BASIC.bizEx(getCacheTag(), bizException, extraInfo, args);}static void logBizEx(BizExceptionMark bizException, String extraInfo, Object... args) {helperLog(() -> log.warn(bizEx(bizException, extraInfo, args)));}/*** 运行时异常tag日志内容生成* <b style="color:red">只有缓存了标签之后才可用</b>*/static String otherEx(Exception e) {return BASIC.otherEx(getCacheTag(), e);}static void logOtherEx(Exception e) {helperLog(() -> log.error(otherEx(e)));}/*** 运行时异常tag日志内容生成* <b style="color:red">只有缓存了标签之后才可用</b>*/static String otherEx(Exception e, String extraInfo, Object... args) {return BASIC.otherEx(getCacheTag(), e, extraInfo, args);}static void logOtherEx(Exception e, String extraInfo, Object... args) {helperLog(() -> log.error(otherEx(e, extraInfo, args)));}/*** 通用标签日志包装<br/>* <b style="color:red">只有缓存了标签之后才可用</b>** @param runnable 你要执行的无返回值方法*/static void catchLog(Runnable runnable) {try {runnable.run();} catch (Exception e) {if (e instanceof BizExceptionMark) {helperLog(() -> log.warn(bizEx((BizExceptionMark) e)));} else {helperLog(() -> log.error(otherEx(e)));}}}/*** 通用标签日志包装<br/>* <b style="color:red">只有缓存了标签之后才可用</b>** @param runnable 你要执行的无返回值方法*/static void catchLog(Runnable runnable, String extraInfo, Object... args) {try {runnable.run();} catch (Exception e) {if (e instanceof BizExceptionMark) {helperLog(() -> log.warn(bizEx((BizExceptionMark) e, extraInfo, args)));} else {helperLog(() -> log.error(otherEx(e, extraInfo, args)));}}}/*** 通用标签日志包装<br/>* <b style="color:red">只有缓存了标签之后才可用</b>** @param supplier 你要执行的有返回值方法* @return 方法执行结果(可能为null)*/static <R> @Nullable R catchLog(Supplier<R> supplier) {try {return supplier.get();} catch (Exception e) {if (e instanceof BizExceptionMark) {helperLog(() -> log.warn(bizEx((BizExceptionMark) e)));} else {helperLog(() -> log.error(otherEx(e)));}}return null;}/*** 通用标签日志包装<br/>* <b style="color:red">只有缓存了标签之后才可用</b>** @param supplier 你要执行的有返回值方法* @return 方法执行结果(可能为null)*/static <R> @Nullable R catchLog(Supplier<R> supplier, String extraInfo, Object... args) {try {return supplier.get();} catch (Exception e) {if (e instanceof BizExceptionMark) {helperLog(() -> log.warn(bizEx((BizExceptionMark) e, extraInfo, args)));} else {helperLog(() -> log.error(otherEx(e, extraInfo, args)));}}return null;}}
}

相关博客

  • Spring实现Logback日志模板设置动态参数
  • Spring实现输出带动态标签的日志

http://www.ppmy.cn/devtools/150037.html

相关文章

【线性代数】通俗理解特征向量与特征值

这一块在线性代数中属于重点且较难理解的内容&#xff0c;下面仅个人学习过程中的体会&#xff0c;错误之处欢迎指出&#xff0c;有更简洁易懂的理解方式也欢迎留言学习。 文章目录 概念计算几何直观理解意义PS.适用 概念 矩阵本身就是一个线性变换&#xff0c;对一个空间中的…

为AI聊天工具添加一个知识系统 开发环境准备

现在&#xff0c;我准备开始开发这个项目&#xff0c;需要搭建开发环境 并将前面的程序整理到项目文件中。请完成--我是一个新手 好的&#xff01;我将帮助您从头开始搭建开发环境&#xff0c;并整理好之前的程序代码到项目文件中&#xff0c;以便您可以轻松启动这个项目。以下…

模式识别-Ch3-贝叶斯估计

贝叶斯估计 贝叶斯估计是概率密度估计中另一类主要的参数估计方法。其结果在很多情况下与最大似然法十分相似&#xff0c;但是&#xff0c;两种方法对问题的处理视角是不一样的。 贝叶斯估计最大似然估计将待估计的参数视为一个随机变量&#xff0c;其中的一个核心任务是根据…

【每日学点鸿蒙知识】关于热修复、图片预览、多个@State刷新性能问题等

1、是否推荐使用bm quickfix制造修复包&#xff1f; 官方文档文档中显示&#xff1a;快速修复补丁安装bm quickfix -a -f /data/app/有两个问题&#xff1a; hqf文件如何制作的文档没有找到。hqf 是不是新版本和旧版本的差分包咨询场景描述&#xff1a;app可以在运行过程中修…

Docker Compose 教程

Docker Compose 是一个 Docker 容器的依赖管理工具。 例如我们一个服务需要依赖到多个 Docker 容器&#xff0c;那么使用 Docker Compose 这个工具就能很方便的帮助我们管理。 Docker Compose 通过配置文件 .yml。 定义了所有容器的依赖关系。 然后我们只需把我们想要的 Docke…

深度学习中的常见初始化方法:原理、应用与比较

【 Transformer 系列&#xff0c;故事从 d k \sqrt{d_k} dk​ ​说起】 LLM这么火&#xff0c;Transformer厥功甚伟&#xff0c;某天心血来潮~&#xff0c;再去看看&#xff01; 它长这个样子&#xff1a; 深入浅出 Transformer 看完后&#xff0c;想起了老生常谈 d k \sqrt{d_…

windows servre 2008 加密本地windows 操作系统驱动器,bitlocker加解密过程

首先点击服务器管理器&#xff0c;点击功能选项&#xff0c;添加功能 下一步 安装 需要重启 发现安装成功 发现只有一个盘&#xff0c;再添加一个盘 点击虚拟机设置 点击下一步 下一步 下一步 储存为单个文件 点击完成 找到这里 如果脱机就选择联机 右键初始化磁盘 点击确定 右…

uniapp中修改input里的字体颜色

<input type"text" placeholder"从哪儿出发" v-model"start.site_name"placeholder-class"shi-co" disabled"true" value"" /> 给input加上placeholder-class属性&#xff0c;然后给此属性加上样式。 未选…