当前位置: 首页 > news >正文

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文件内具体方法和行号

最近琢磨怎么把日志打的更清晰&#xff0c;方便查找问题&#xff0c;又不需要在每个class内都创建Logger对象&#xff0c;还带上不同的颜色做区分&#xff0c;简直不要太爽。利用堆栈的方向顺序拿到日志的class问题。看效果&#xff0c;直接上代码。 1、demo test 2、输出效果…...

02. Hibernate 初体验之持久化对象

1. 前言 本节课程让我们一起体验 Hibernate 的魅力&#xff01;编写第一个基于 Hibernate 的实例程序。 在本节课程中&#xff0c;你将学到 &#xff1a; Hibernate 的版本发展史&#xff1b;持久化对象的特点。 为了更好地讲解这个内容&#xff0c;这个初体验案例分上下 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包&#xff0c;⽽不⽤⼿动去添加jar包 主要功能 项⽬构建管理依赖 构建Maven项目 1.1项目构建 Maven 提供了标准的,跨平台(Linux, Windows, MacOS等)的⾃动化项⽬构建⽅式 当我们开发了⼀个项⽬之后, 代…...

Qt之元对象系统

Qt的元对象系统提供了信号和槽机制&#xff08;用于对象间的通信&#xff09;、运行时类型信息和动态属性系统。 元对象系统基于三个要素&#xff1a; 1、QObject类为那些可以利用元对象系统的对象提供了一个基类。 2、在类声明中使用Q_OBJECT宏用于启用元对象特性&#xff0c…...

Provider(1)- 什么是AudioBufferProvider

什么是AudioBufferProvider&#xff1f; 顾名思义&#xff0c;Audio音频数据缓冲提供&#xff0c;就是提供音频数据的缓冲类&#xff0c;而且这个AudioBufferProvider派生出许多子类&#xff0c;每个子类有不同的用途&#xff0c;至关重要&#xff1b;那它在Android哪个地方使…...

加密与安全_密钥体系的三个核心目标之完整性解决方案

文章目录 Pre机密性完整性1. 哈希函数&#xff08;Hash Function&#xff09;定义特征常见算法应用散列函数常用场景散列函数无法解决的问题 2. 消息认证码&#xff08;MAC&#xff09;概述定义常见算法工作原理如何使用 MACMAC 的问题 不可否认性数字签名&#xff08;Digital …...

【C++】:继承[下篇](友元静态成员菱形继承菱形虚拟继承)

目录 一&#xff0c;继承与友元二&#xff0c;继承与静态成员三&#xff0c;复杂的菱形继承及菱形虚拟继承四&#xff0c;继承的总结和反思 点击跳转上一篇文章&#xff1a; 【C】&#xff1a;继承(定义&&赋值兼容转换&&作用域&&派生类的默认成员函数…...

昇思25天学习打卡营第13天|基于MindNLP+MusicGen生成自己的个性化音乐

关于MindNLP MindNLP是一个依赖昇思MindSpore向上生长的NLP&#xff08;自然语言处理&#xff09;框架&#xff0c;旨在利用MindSpore的优势特性&#xff0c;如函数式融合编程、动态图功能、数据处理引擎等&#xff0c;致力于提供高效、易用的NLP解决方案。通过全面拥抱Huggin…...

nigix的下载使用

1、官网&#xff1a;https://nginx.org/en/download.html 双击打开 nginx的默认端口是80 配置文件 默认访问页面 在目录下新建pages&#xff0c;放入图片 在浏览器中输入地址进行访问 可以在电脑中配置本地域名 Windows设置本地DNS域名解析hosts文件配置 文件地址&#xf…...

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&#xff1a; 图例&#xff1a; 动态规划思路 2 代码实现&#xff1a; 3 空间优化&#xff1a; 代码实现&#xff1a; 下面是0-1背包和完全背包具体的例题&#xff1a; 代码实现&#xff1a; 图例&#xff1a; 空间优化代码示例 Question&#xff1a; 给定n个物品…...

C++catch (...)陈述

catch (...)陈述 例外处理可以有多个catch&#xff0c;如果catch后的小括弧里面放...&#xff0c;就表示不限型态种类的任何例外。 举例如下 #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时&#xff0c;发现访问延迟突然增大&#xff0c;如何进行排查&#xff1f; 首先&#xff0c;第一步&#xff0c;建议你去查看一下Redis的慢日志。Redis提供了慢日志命令的统计功能&#xff0c;我们通过以下设置&#xff0c;就…...

【Lora模型推荐】Stable Diffusion创作具有玉石翡翠质感的图标设计

站长素材AI教程是站长之家旗下AI绘图教程平台 海量AI免费教程&#xff0c;每日更新干货内容 想要深入学习更多AI绘图教程&#xff0c;请访问站长素材AI教程网&#xff1a; AI教程_深度学习入门指南 - 站长素材 (chinaz.com) logo版权归各公司所有&#xff01;本笔记仅供AIGC…...

vscode 远程开发

目录 vscode 远程连接 选择 Python 环境 vscode 远程连接 按 CtrlShiftP 打开命令面板。输入并选择 Remote-SSH: Open SSH Configuration File...。选择 ~/.ssh/config 文件&#xff08;如果有多个选项&#xff09;。在打开的文件中添加或修改你的 SSH 配置。 这个可以右键…...

前端Vue组件化实践:打造灵活可维护的地址管理组件

随着前端技术的不断演进&#xff0c;复杂度和开发难度也随之上升。传统的一体化开发模式使得每次小小的修改或功能增加都可能牵一发而动全身&#xff0c;严重影响了开发效率和维护成本。组件化开发作为一种解决方案&#xff0c;通过模块化、独立化的开发方式&#xff0c;实现了…...

虚幻引擎ue5游戏运行界面白茫茫一片,怎么处理

根剧下图顺序即可调节游戏运行界面光照问题&#xff1a; 在大纲里找到post&#xff0c;然后选中它&#xff0c;找到Exposure 把最低亮度和最高亮度的0改为1即可...

《代理选择与反爬虫策略探究:如何优化网络爬虫效率与稳定性》

代理IP如何选以及常见反爬策略 为什么需要代理&#xff1f; 因为有的网站会封IP&#xff0c;用户如果没有登录&#xff0c;那IP就是身份标识&#xff0c;如果网站发现用户行为异常就非常可能封IP 什么是代理IP 就是让一个人帮你转交请求&#xff0c;帮你转交的人对面不熟&a…...

Kotlin Flow 防抖 节流

防抖和节流是针对响应跟不上触发频率这类问题的两种解决方案。 一:防抖&#xff08;debounce&#xff09;的概念&#xff1a; 防抖是指当持续触发事件时&#xff0c;一定时间段内没有再触发事件&#xff0c;事件处理函数才会执行一次&#xff0c; 如果设定时间到来之前&#x…...

Android Studio下载与安装

Android Studio下载与安装_android studio下载安装-CSDN博客...

【LC刷题】DAY24:122 55 45 1005

122. 买卖股票的最佳时机 II class Solution { public:int maxProfit(vector<int>& prices) {int result 0;for(int i 1; i < prices.size(); i ){result max(prices[i] - prices[ i - 1], 0);}return result;} };55. 跳跃游戏 link class Solution { public…...

从零开始的python学习生活2

接上封装 class Phone:__volt0.5def __keepsinglecore(self):print("让cpu以单核运行")def if5G(self):if self.__volt>1:print("5G通话已开启")else:self.__keepsinglecore()print("电量不足&#xff0c;无法使用5G通话&#xff0c;已经设置为单…...

【并发编程】进程 线程 协程

进程&#xff08;Process&#xff09;、线程&#xff08;Thread&#xff09;和协程&#xff08;Coroutine&#xff09;构成了计算机科学中实现任务并发执行的三种核心抽象机制。通常&#xff0c;为了提高程序的执行效率&#xff0c;开发者会根据应用场景和性能需求&#xff0c;…...

Vue的生命周期函数有哪些?详细说明

Vue.js 的生命周期函数包括以下几个阶段&#xff0c;每个阶段都有相应的钩子函数可以用来在特定时机执行自定义的逻辑。这些生命周期钩子函数使得我们可以在组件的不同阶段进行操作&#xff0c;从而管理组件的状态和行为。 1. beforeCreate&#xff1a; - 描述&#xff1a;…...

大语言模型应用--AI工程化落地

文章目录 大语言模型概述什么是大语言模型什么是机器学习什么是深度学习 理解大语言模型历史沿革关键 AIGC系统AI工程化项目的落地落地的方法Prompt工程&#xff08;第一阶段&#xff09;RAG检索&#xff08;第二阶段&#xff09;训练特定功能模型&#xff08;第三阶段&#xf…...

我会什么开发技能

java我会什么&#xff1f; 一、并发编程 1、并发编程&#xff1a;jdk中的courren包只能够类实现&#xff08;seamplore&#xff0c;CountDownLaunch&#xff0c;Pharse&#xff0c;CycliBarrier&#xff0c;CompletableFuture&#xff09;&#xff0c;AQS的原理&#xff0c;线…...

Run LoongArch64 Alpine VM on x86_64

一、Build from source(build on x86_64) Obtain the latest libvirt, virt-manager, and qemu source code, compile and install them. 1.1 Build libvirt from source sudo apt-get update sudo apt-get install augeas-tools bash-completion debhelper-compat dh-apparm…...

4层负载均衡和7层负载均衡

四层负载均衡&#xff08;Layer 4 Load Balancing&#xff09;指的是在网络传输层&#xff08;TCP/IP模型中的第四层&#xff09;进行负载均衡的技术。四层负载均衡通常使用IP地址、端口号和协议等信息来将网络流量分配到多个服务器上。它主要关心网络层的信息&#xff0c;不涉…...

前端Vue组件化实践:打造仿京东天猫商品属性选择器组件

在前端开发领域&#xff0c;随着业务需求的日益复杂和技术的不断进步&#xff0c;传统的整体式应用开发模式已逐渐显得捉襟见肘。面对日益庞大的系统&#xff0c;每次微小的功能修改或增加都可能导致整个逻辑结构的重构&#xff0c;形成牵一发而动全身的困境。为了解决这一问题…...

沈阳建网站企业/怎样做一个网站

Link&#xff1a; 原型模型设计工具&#xff1a;墨刀。&#xff08;在学长的推荐下发现有这一款十分好用的原型模型设计工具呢。^_^.&#xff09; 小伙伴链接&#xff1a;http://www.cnblogs.com/mingsonic/p/9683453.html 原型链接&#xff1a;&#xff08;如果小樱同学看不懂…...

网站设计开发软件有哪些/新闻稿在线

QT怎么自动一次性修改、替换程序中所有变量名? 熟悉的编译器都有这个功能&#xff0c; 初学QT&#xff0c;搜索无果&#xff0c;找了一会儿&#xff0c;分析给大家。 最直接的办法&#xff1a;选中变量名&#xff0c;CtrlShiftr &#xff0c;变成红色之后即可更改。. 选中变…...

泰安北京网站建设/seo推广是做什么的

ikvmc 将 Java 字节码转换成 .NET dll 文件或是 .exe 文件。 本文内容 用法选项备注 例子用法 ikvmc [ options ] classOrJarfile [ classOrJarfile ... ] 其中&#xff0c; options 将在下面介绍。classOrJarfile 是 Java .class 文件或 jar 文件&#xff0c;可以包含通配符&…...

手机网站重要性/网站优化资源

小红书类种草文标题的写法分享 请看下图&#xff1a;...

我想做网站卖衣服做/制作一个网站的费用是多少

问题描述 农田灌溉是一项十分费体力的农活&#xff0c;特别是大型的农田。蒜头君想为农民伯伯们减轻农作负担&#xff0c;最近在研究一款高科技——灌溉机器人。它可以在远程电脑控制下&#xff0c;给农田里的作物进行灌溉。现在有一片 N 行 M 列的农田。农田的土壤有两种类型&…...

关于做网站ppt/外贸网站有哪些

要写出高效的SQL&#xff0c;那么必须必须得清楚SQL执行路径&#xff0c;介绍如何提高SQL性能的文章很多&#xff0c;这里不再赘述&#xff0c;本人来谈谈如何从 减少SQL回表次数 来提高查询性能&#xff0c;因为回表将导致扫描更多的数据块。 我们大家都知道&#xff0c;数据库…...