java LogUtil输出日志打日志的class文件内具体方法和行号
最近琢磨怎么把日志打的更清晰,方便查找问题,又不需要在每个class内都创建Logger对象,还带上不同的颜色做区分,简直不要太爽。利用堆栈的方向顺序拿到日志的class问题。看效果,直接上代码。
1、demo test

2、输出效果

3、完整的LogUtil文件
>>>>>>>> java 版本代码 :
import org.jetbrains.annotations.NotNull;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;public class LogUtil {public static void trace(String msg) {getLogger().trace(msg);}public static void debug(String msg) {getLogger().debug("\u001B[34m" + msg + "\u001B[0m");}public static void info(String msg) {getLogger().info("\u001B[32m" + msg + "\u001B[0m");}public static void warning(String msg) {getLogger().warn("\u001B[33m" + msg + "\u001B[0m");}public static void error(String msg) {getLogger().error("\u001B[31m" + msg + "\u001B[0m");}public static void error(String msg, Throwable t) {getLogger().error("\u001B[31m" + msg + "\u001B[0m", t);}private static Logger getLogger() {return LoggerFactory.getLogger(findCaller());}/*** 查找调用者的方法。* 通过分析当前线程的堆栈跟踪,找到第一个不是LogUtil类的方法作为调用者。* 这个方法主要用于在日志记录时,确定日志的来源,以便更好地定位问题。** @return 调用者的类路径和方法名。*/private static String findCaller() {// 获取当前线程的堆栈跟踪元素// 获取堆栈信息StackTraceElement[] callStack = Thread.currentThread().getStackTrace();// 初始化调用者堆栈元素为null// 最原始被调用的堆栈信息StackTraceElement caller = null;// 获取LogUtil类的全限定名// 日志类名称String logClassName = LogUtil.class.getName();// 遍历堆栈跟踪元素,找到第一个不是LogUtil类的元素// 循环遍历到日志类标识int i = 0;for (int len = callStack.length; i < len; i++) {if (logClassName.equals(callStack[i].getClassName())) {break;}}// 调用者位于LogUtil类元素之后的第三个元素(考虑到findCaller自身和getStackTrace的调用)caller = callStack[i + 3];// 获取调用者的类路径和方法名String fullPath = getPath(caller);// 返回调用者的类路径和方法名return fullPath;}/*** 获取调用者堆栈跟踪元素的详细路径。* <p>* 本方法用于生成调用者方法的详细路径,包括类名、方法名和行号。这样生成的路径可以帮助定位代码中调用此方法的位置,* 对于调试和跟踪代码执行流程非常有用。** @param caller 调用者堆栈跟踪元素,通过StackTraceElement获取。* @return 返回一个字符串,包含调用者类名、方法名和行号的详细路径。*/@NotNullprivate static String getPath(StackTraceElement caller) {// 格式化类名,去除包名,只保留类名部分。String formatClassName = caller.getClassName();// 获取调用者方法名。String methodName = caller.getMethodName();// 获取调用者方法所在的行号。int lineNumber = caller.getLineNumber();// 拼接类名、方法名和行号,生成详细的路径。String fullPath = formatClassName + "." + methodName + "." + lineNumber;return fullPath;}/*** 格式化类名。* 将带有包名的类名转换为首字母大写的形式,去除包名并连接类名。* 如果类名本身不包含包名,直接返回原类名。* @param name 完整的类名,包括包名。* @return 格式化后的类名,不包含包名。*/private static String simpleClassName(String name) {// 使用正则表达式按点号分割类名和包名String[] split = name.split("\\.");// 使用 StringBuffer 来构建最终的类名,以确保线程安全和效率StringBuffer buf = new StringBuffer();for (int i = 0; i < split.length; i++) {// 如果当前元素不是最后一个元素,说明它是包名的一部分// 取出每个包名部分的第二个字符(即首字母)并连接到缓冲区// 如果是最后一个元素,直接添加到缓冲区,以保留类名的完整性if (i != split.length - 1) {buf.append(split[i].charAt(1));buf.append(".");} else {buf.append(split[i]);}}// 将缓冲区转换为字符串并返回return buf.toString();}}
>>>>>>>> scala 版本代码 :
import org.apache.log4j.Logger
import org.apache.log4j.LogManager
import scala.util.control.Breaks.{break, breakable}class LOGGER() {}/*** Created by wqj on 16/6/30.* 日志记录接口,跟spark本身的日志分开记录*/
object LOGGER extends Serializable {/*** 输出debug日志** @param message ,日志消息对象* */def debug(message: Object): Unit = {getLogger.debug("\u001B[34m" + message + "\u001B[0m")}/*** 输出info日志** @param message ,日志消息对象* */def file(message: Object): Unit = {getLogger.info(message)}/*** 输出warn日志** @param message ,日志消息对象* */def warn(message: Object): Unit = {getLogger.warn("\u001B[33m" + message + "\u001B[0m")}/*** 输出error日志** @param message ,日志消息对象* */def error(message: Object): Unit = {getLogger.error("\u001B[31m" + message + "\u001B[0m")}/*** 输出info日志, 由于本处打印的日志无法与spark日志分开, 故使用error等级输出属于info范畴的日志. 此处使用logInfo间接调用error, 为* 以后可能实现日志分开做准备** @param message ,日志消息对象* */def info(message: Object): Unit = {getLogger.info("\u001B[32m" + message + "\u001B[0m")}/*** 获取日志记录器** @return 日志记录器* */def getLogger(): Logger = {LogManager.getLogger(findCaller)}/*** 查找调用者的方法。* 通过分析当前线程的堆栈跟踪,找到第一个不是LogUtil类的方法作为调用者。* 这个方法主要用于在日志记录时,确定日志的来源,以便更好地定位问题。** @return 调用者的类路径和方法名。*/private def findCaller(): String = { // 获取当前线程的堆栈跟踪元素// 获取堆栈信息val callStack = Thread.currentThread.getStackTrace// 初始化调用者堆栈元素为null// 最原始被调用的堆栈信息var caller: StackTraceElement = null// 获取LogUtil类的全限定名// 日志类名称val logClassName = LOGGER.getClass.getName// 遍历堆栈跟踪元素,找到第一个不是LogUtil类的元素// 循环遍历到日志类标识var i = 0val len = callStack.lengthbreakable {while (i < len) {if (logClassName == callStack(i).getClassName) breaki += 1}}val index = i + 3// 调用者位于LogUtil类元素之后的第三个元素(考虑到findCaller自身和getStackTrace的调用)caller = callStack(index)// 获取调用者的类路径和方法名val fullPath = getFullPath(caller)// 返回调用者的类路径和方法名fullPath}/*** 获取调用者堆栈跟踪元素的详细路径。* <p>* 本方法用于生成调用者方法的详细路径,包括类名、方法名和行号。这样生成的路径可以帮助定位代码中调用此方法的位置,* 对于调试和跟踪代码执行流程非常有用。** @param caller 调用者堆栈跟踪元素,通过StackTraceElement获取。* @return 返回一个字符串,包含调用者类名、方法名和行号的详细路径。*/private def getFullPath(caller: StackTraceElement): String = { // 格式化类名,去除包名,只保留类名部分。val formatClassName = caller.getClassName// 获取调用者方法名。val methodName = caller.getMethodName// 获取调用者方法所在的行号。val lineNumber = caller.getLineNumber// 拼接类名、方法名和行号,生成详细的路径。formatClassName + "." + methodName + "." + lineNumber}}
4、springboot 日志收集配置logback-spring.xml文件,这个文件根据需求编辑
<?xml version="1.0" encoding="UTF-8"?>
<configuration scan="ture" scanPeriod="60 seconds" debug="false"><!-- 定义日志的根目录 --><property name="LOG_HOME" value="./logs"/><!-- 定义日志文件名称 --><property name="APP_NAME" value="myApp"/><!-- 控制台输出 --><appender name="console" class="ch.qos.logback.core.ConsoleAppender"><filter class="ch.qos.logback.classic.filter.ThresholdFilter"><level>INFO</level></filter><encoder><charset>UTF-8</charset><!-- <pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50} - %msg%n</pattern>--><pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %highlight(%-5level) %green(%logger{50}) - %msg%n</pattern></encoder></appender><!-- 滚动记录文件,先将日志记录到指定文件,当符合某个条件时,将日志记录到其他文件 --><appender name="appLogAppender" class="ch.qos.logback.core.rolling.RollingFileAppender"><!-- 指定日志文件的名称 --><file>${LOG_HOME}/${APP_NAME}.log</file><!-- 当发生滚动时,决定 RollingFileAppender 的行为,涉及文件移动和重命名 --><rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"><!-- 滚动时产生的文件的存放位置及文件名称 %d{yyyy-MM-dd}:按天进行日志滚动 --><fileNamePattern>${LOG_HOME}/${APP_NAME}-%d{yyyy-MM-dd}.log</fileNamePattern><!-- 只保存最近30天的文件 --><MaxHistory>30</MaxHistory><!--用来指定日志文件的上限大小 --><totalSizeCap>10GB</totalSizeCap>--></rollingPolicy><!-- 日志输出编码格式化 --><encoder><charset>UTF-8</charset><pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50}:%line - %msg%n</pattern></encoder></appender><root level="INFO"><appender-ref ref="console"/><appender-ref ref="appLogAppender"/></root>
</configuration>
5、springboot项目log4j配置,这个文件可以自定义,主要是要获取 %c: 输出日志的logger名, 即Logger.getLogger中的名字或全限定类名
log4j.rootLogger=DEBUG, stdout, R
log4j.appender.stdout=org.apache.log4j.ConsoleAppender
log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern=%d{yyyy-MM-dd HH:mm:ss} [%t] %-5p %c - %m%nlog4j.appender.R=org.apache.log4j.RollingFileAppender
log4j.appender.R.File=ai.log
log4j.appender.R.MaxFileSize=10240KB
log4j.appender.R.MaxBackupIndex=10
log4j.appender.R.layout=org.apache.log4j.PatternLayout
log4j.appender.R.layout.ConversionPattern=%d{yyyy-MM-dd HH:mm:ss} [%t] %-5p %c - %m%nlog4j.logger.org.springframework=WARN
log4j.logger.com.lagooo.as=INFO
log4j.logger.org.apache=INFO
log4j.logger.org.apache.log4j.Logger=error相关文章:
java LogUtil输出日志打日志的class文件内具体方法和行号
最近琢磨怎么把日志打的更清晰,方便查找问题,又不需要在每个class内都创建Logger对象,还带上不同的颜色做区分,简直不要太爽。利用堆栈的方向顺序拿到日志的class问题。看效果,直接上代码。 1、demo test 2、输出效果…...
02. Hibernate 初体验之持久化对象
1. 前言 本节课程让我们一起体验 Hibernate 的魅力!编写第一个基于 Hibernate 的实例程序。 在本节课程中,你将学到 : Hibernate 的版本发展史;持久化对象的特点。 为了更好地讲解这个内容,这个初体验案例分上下 2…...
MySQL超详细学习教程,2023年硬核学习路线
文章目录 前言1. 数据库的相关概念1.1 数据1.2 数据库1.3 数据库管理系统1.4 数据库系统1.5 SQL 2. MySQL数据库2.1 MySQL安装2.2 MySQL配置2.2.1 添加环境变量2.2.2 新建配置文件2.2.3 初始化MySQL2.2.4 注册MySQL服务2.2.5 启动MySQL服务 2.3 MySQL登录和退出2.4 MySQL卸载2.…...
初识SpringBoot
1.Maven Maven是⼀个项⽬管理⼯具, 通过pom.xml⽂件的配置获取jar包,⽽不⽤⼿动去添加jar包 主要功能 项⽬构建管理依赖 构建Maven项目 1.1项目构建 Maven 提供了标准的,跨平台(Linux, Windows, MacOS等)的⾃动化项⽬构建⽅式 当我们开发了⼀个项⽬之后, 代…...
Qt之元对象系统
Qt的元对象系统提供了信号和槽机制(用于对象间的通信)、运行时类型信息和动态属性系统。 元对象系统基于三个要素: 1、QObject类为那些可以利用元对象系统的对象提供了一个基类。 2、在类声明中使用Q_OBJECT宏用于启用元对象特性,…...
Provider(1)- 什么是AudioBufferProvider
什么是AudioBufferProvider? 顾名思义,Audio音频数据缓冲提供,就是提供音频数据的缓冲类,而且这个AudioBufferProvider派生出许多子类,每个子类有不同的用途,至关重要;那它在Android哪个地方使…...
加密与安全_密钥体系的三个核心目标之完整性解决方案
文章目录 Pre机密性完整性1. 哈希函数(Hash Function)定义特征常见算法应用散列函数常用场景散列函数无法解决的问题 2. 消息认证码(MAC)概述定义常见算法工作原理如何使用 MACMAC 的问题 不可否认性数字签名(Digital …...
【C++】:继承[下篇](友元静态成员菱形继承菱形虚拟继承)
目录 一,继承与友元二,继承与静态成员三,复杂的菱形继承及菱形虚拟继承四,继承的总结和反思 点击跳转上一篇文章: 【C】:继承(定义&&赋值兼容转换&&作用域&&派生类的默认成员函数…...
昇思25天学习打卡营第13天|基于MindNLP+MusicGen生成自己的个性化音乐
关于MindNLP MindNLP是一个依赖昇思MindSpore向上生长的NLP(自然语言处理)框架,旨在利用MindSpore的优势特性,如函数式融合编程、动态图功能、数据处理引擎等,致力于提供高效、易用的NLP解决方案。通过全面拥抱Huggin…...
nigix的下载使用
1、官网:https://nginx.org/en/download.html 双击打开 nginx的默认端口是80 配置文件 默认访问页面 在目录下新建pages,放入图片 在浏览器中输入地址进行访问 可以在电脑中配置本地域名 Windows设置本地DNS域名解析hosts文件配置 文件地址…...
nginx+lua 实现URL重定向(根据传入的参数条件)
程序版本说明 程序版本URLnginx1.27.0https://nginx.org/download/nginx-1.27.0.tar.gzngx_devel_kitv0.3.3https://github.com/simpl/ngx_devel_kit/archive/v0.3.3.tar.gzluajitv2.1https://github.com/openresty/luajit2/archive/refs/tags/v2.1-20240626.tar.gzlua-nginx-m…...
算法学习笔记(8.4)-完全背包问题
目录 Question: 图例: 动态规划思路 2 代码实现: 3 空间优化: 代码实现: 下面是0-1背包和完全背包具体的例题: 代码实现: 图例: 空间优化代码示例 Question: 给定n个物品…...
C++catch (...)陈述
catch (...)陈述 例外处理可以有多个catch,如果catch后的小括弧里面放...,就表示不限型态种类的任何例外。 举例如下 #include <iostream>int main() {int i -1;try {if (i > 0) {throw 0;}throw 2.0;}catch (const int e) {std::cout <…...
Redis实践
Redis实践 使用复杂度高的命令 如果在使用Redis时,发现访问延迟突然增大,如何进行排查? 首先,第一步,建议你去查看一下Redis的慢日志。Redis提供了慢日志命令的统计功能,我们通过以下设置,就…...
【Lora模型推荐】Stable Diffusion创作具有玉石翡翠质感的图标设计
站长素材AI教程是站长之家旗下AI绘图教程平台 海量AI免费教程,每日更新干货内容 想要深入学习更多AI绘图教程,请访问站长素材AI教程网: AI教程_深度学习入门指南 - 站长素材 (chinaz.com) logo版权归各公司所有!本笔记仅供AIGC…...
vscode 远程开发
目录 vscode 远程连接 选择 Python 环境 vscode 远程连接 按 CtrlShiftP 打开命令面板。输入并选择 Remote-SSH: Open SSH Configuration File...。选择 ~/.ssh/config 文件(如果有多个选项)。在打开的文件中添加或修改你的 SSH 配置。 这个可以右键…...
前端Vue组件化实践:打造灵活可维护的地址管理组件
随着前端技术的不断演进,复杂度和开发难度也随之上升。传统的一体化开发模式使得每次小小的修改或功能增加都可能牵一发而动全身,严重影响了开发效率和维护成本。组件化开发作为一种解决方案,通过模块化、独立化的开发方式,实现了…...
虚幻引擎ue5游戏运行界面白茫茫一片,怎么处理
根剧下图顺序即可调节游戏运行界面光照问题: 在大纲里找到post,然后选中它,找到Exposure 把最低亮度和最高亮度的0改为1即可...
《代理选择与反爬虫策略探究:如何优化网络爬虫效率与稳定性》
代理IP如何选以及常见反爬策略 为什么需要代理? 因为有的网站会封IP,用户如果没有登录,那IP就是身份标识,如果网站发现用户行为异常就非常可能封IP 什么是代理IP 就是让一个人帮你转交请求,帮你转交的人对面不熟&a…...
Kotlin Flow 防抖 节流
防抖和节流是针对响应跟不上触发频率这类问题的两种解决方案。 一:防抖(debounce)的概念: 防抖是指当持续触发事件时,一定时间段内没有再触发事件,事件处理函数才会执行一次, 如果设定时间到来之前&#x…...
网络六边形受到攻击
大家读完觉得有帮助记得关注和点赞!!! 抽象 现代智能交通系统 (ITS) 的一个关键要求是能够以安全、可靠和匿名的方式从互联车辆和移动设备收集地理参考数据。Nexagon 协议建立在 IETF 定位器/ID 分离协议 (…...
装饰模式(Decorator Pattern)重构java邮件发奖系统实战
前言 现在我们有个如下的需求,设计一个邮件发奖的小系统, 需求 1.数据验证 → 2. 敏感信息加密 → 3. 日志记录 → 4. 实际发送邮件 装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其…...
ardupilot 开发环境eclipse 中import 缺少C++
目录 文章目录 目录摘要1.修复过程摘要 本节主要解决ardupilot 开发环境eclipse 中import 缺少C++,无法导入ardupilot代码,会引起查看不方便的问题。如下图所示 1.修复过程 0.安装ubuntu 软件中自带的eclipse 1.打开eclipse—Help—install new software 2.在 Work with中…...
Typeerror: cannot read properties of undefined (reading ‘XXX‘)
最近需要在离线机器上运行软件,所以得把软件用docker打包起来,大部分功能都没问题,出了一个奇怪的事情。同样的代码,在本机上用vscode可以运行起来,但是打包之后在docker里出现了问题。使用的是dialog组件,…...
HashMap中的put方法执行流程(流程图)
1 put操作整体流程 HashMap 的 put 操作是其最核心的功能之一。在 JDK 1.8 及以后版本中,其主要逻辑封装在 putVal 这个内部方法中。整个过程大致如下: 初始判断与哈希计算: 首先,putVal 方法会检查当前的 table(也就…...
HarmonyOS运动开发:如何用mpchart绘制运动配速图表
##鸿蒙核心技术##运动开发##Sensor Service Kit(传感器服务)# 前言 在运动类应用中,运动数据的可视化是提升用户体验的重要环节。通过直观的图表展示运动过程中的关键数据,如配速、距离、卡路里消耗等,用户可以更清晰…...
Redis上篇--知识点总结
Redis上篇–解析 本文大部分知识整理自网上,在正文结束后都会附上参考地址。如果想要深入或者详细学习可以通过文末链接跳转学习。 1. 基本介绍 Redis 是一个开源的、高性能的 内存键值数据库,Redis 的键值对中的 key 就是字符串对象,而 val…...
LeetCode 0386.字典序排数:细心总结条件
【LetMeFly】386.字典序排数:细心总结条件 力扣题目链接:https://leetcode.cn/problems/lexicographical-numbers/ 给你一个整数 n ,按字典序返回范围 [1, n] 内所有整数。 你必须设计一个时间复杂度为 O(n) 且使用 O(1) 额外空间的算法。…...
篇章一 论坛系统——前置知识
目录 1.软件开发 1.1 软件的生命周期 1.2 面向对象 1.3 CS、BS架构 1.CS架构编辑 2.BS架构 1.4 软件需求 1.需求分类 2.需求获取 1.5 需求分析 1. 工作内容 1.6 面向对象分析 1.OOA的任务 2.统一建模语言UML 3. 用例模型 3.1 用例图的元素 3.2 建立用例模型 …...
Ubantu-Docker配置最新镜像源250605
尝试其他镜像加速器 阿里云镜像加速器:登录阿里云,进入容器镜像服务获取专属加速器地址。毫秒镜像:https://docker.1ms.run。DockerHub镜像加速器:https://docker.xuanyuan.me。Docker Hub 镜像加速服务:https://dock…...
