日志框架LOG4J2系列六——log4j2使用包装器

本文旨在解决使用log4j2使用包装器时,不能打印正确行号问题

slf4j+log4j2组合使用时,有时会使用包装器LoggerWrapper(装饰器LoggerDecorator)对原生的Logger能力进行增强,如修改日志的入参或对日志增加一些定位信息。如对日志的输入增加统一的前缀logPrefix

public class LoggerWrapper implements Logger {private final Logger logger;private final String logPrefix;public LoggerWrapper(String logPrefix, Logger logger) {this.logPrefix = logPrefix;this.logger = logger;}@Overridepublic void info(String message, Object arg) {this.logger.info(logPrefix + message, arg);}// ... 其他日志方法
}

在使用时通过统一的一个工厂内提供LoggerWrapper作为logger使用

public class MyLogger {public Logger logger(Class<?> clazz) {Logger logger = LoggerFactory.getLogger(clazz);// 未业务打印的日志都携带一个 “MyLogger”的标记,与其他包内的日志区分开来return new LoggerWrapper("MyLogger", logger);}
}

在其他业务类使用时可以按如下方式使用

public class MyBussiness {private static final Logger LOGGER = MyLogger.logger(MyBussiness.class);public void doSomething1() {LOGGER.info("MyBussiness doSomething1");}public void doSomething2() {LOGGER.info("MyBussiness doSomething2");}
}

期望日志为打印业务日志的类名与行号,使用log4j2的[%c %L]配置,实际日志输出

[2023-11-01 16:00:00,000 +0800] [INFO ][main][MyBussiness 12] MyBussiness doSomething1
[2023-11-01 16:00:00,000 +0800] [INFO ][main][MyBussiness 12] MyBussiness doSomething2

发现log4j2打印的行号错误,都打印相同的行号,且是LoggerWrapper.info()方法的行数,不是实际业务打印日志的行数。当使用包装器后如果只按上述配置可能会存在此问题。log4j2将自定义的logger也识别为业务代码。

解决此问题的方法是指定log4j2的FQCN。Log4j 会记住 Logger 的全限定类名(FQCN),并在打印位置时使用它在每个日志事件中堆栈进行遍历,打印全限定类名(FQCN)的上一个堆栈的行号。上述LoggerWrapper没有指定FQCN,导致Log4j2认为FQCN是其内部的Logger,他的上一层日志事件是this.logger.info(logPrefix + message, arg);,导致行号永远是LoggerWrapper的内部位置。

正确使用方式应该要实现LocationAwareLogger

public class LoggerWrapper implements LocationAwareLogger {private final LocationAwareLogger logger;private final String logPrefix;// 指定FQCNprivate final String FQCN = LoggerWrapper.class.getName();public LoggerWrapper(String logPrefix, LocationAwareLogger logger) {this.logPrefix = logPrefix;this.logger = logger;}private void writeLog(Marker marker, int level, Throwable exception, String format, Object... args) {String message = format;if (args != null && args.length > 0) {FormattingTuple formatted = MessageFormatter.arrayFormat(format, args);if (exception == null && formatted.getThrowable() != null) {exception = formatted.getThrowable();}message = formatted.getMessage();}// 将FQCN传入Log4j2的logger中logger.log(marker, FQCN, level, addPrefix(message), null, exception);}@Overridepublic void info(String message, Object arg) {writeLog(null, LocationAwareLogger.INFO_INT, null, message, arg);}// ... 其他日志方法
}

此时可以打印正常的日志行号

[2023-11-01 16:00:00,000 +0800] [INFO ][main][MyBussiness 5] MyBussiness doSomething1
[2023-11-01 16:00:00,000 +0800] [INFO ][main][MyBussiness 8] MyBussiness doSomething2

具体实现可以参考kafka-clientLogContext类如下所示:

/** Licensed to the Apache Software Foundation (ASF) under one or more* contributor license agreements. See the NOTICE file distributed with* this work for additional information regarding copyright ownership.* The ASF licenses this file to You under the Apache License, Version 2.0* (the "License"); you may not use this file except in compliance with* the License. You may obtain a copy of the License at**    http://www.apache.org/licenses/LICENSE-2.0** Unless required by applicable law or agreed to in writing, software* distributed under the License is distributed on an "AS IS" BASIS,* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.* See the License for the specific language governing permissions and* limitations under the License.*/
package org.apache.kafka.common.utils;import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.slf4j.Marker;
import org.slf4j.helpers.FormattingTuple;
import org.slf4j.helpers.MessageFormatter;
import org.slf4j.spi.LocationAwareLogger;/*** This class provides a way to instrument loggers with a common context which can be used to* automatically enrich log messages. For example, in the KafkaConsumer, it is often useful to know* the groupId of the consumer, so this can be added to a context object which can then be passed to* all of the dependent components in order to build new loggers. This removes the need to manually* add the groupId to each message.*/
public class LogContext {private final String logPrefix;public LogContext(String logPrefix) {this.logPrefix = logPrefix == null ? "" : logPrefix;}public LogContext() {this("");}public Logger logger(Class<?> clazz) {Logger logger = LoggerFactory.getLogger(clazz);if (logger instanceof LocationAwareLogger) {return new LocationAwareKafkaLogger(logPrefix, (LocationAwareLogger) logger);} else {return new LocationIgnorantKafkaLogger(logPrefix, logger);}}public String logPrefix() {return logPrefix;}private static abstract class AbstractKafkaLogger implements Logger {private final String prefix;protected AbstractKafkaLogger(final String prefix) {this.prefix = prefix;}protected String addPrefix(final String message) {return prefix + message;}}private static class LocationAwareKafkaLogger extends AbstractKafkaLogger {private final LocationAwareLogger logger;private final String fqcn;LocationAwareKafkaLogger(String logPrefix, LocationAwareLogger logger) {super(logPrefix);this.logger = logger;this.fqcn = LocationAwareKafkaLogger.class.getName();}@Overridepublic String getName() {return logger.getName();}@Overridepublic void info(String msg) {writeLog(null, LocationAwareLogger.INFO_INT, msg, null, null);}private void writeLog(Marker marker, int level, String format, Object[] args, Throwable exception) {String message = format;if (args != null && args.length > 0) {FormattingTuple formatted = MessageFormatter.arrayFormat(format, args);if (exception == null && formatted.getThrowable() != null) {exception = formatted.getThrowable();}message = formatted.getMessage();}logger.log(marker, fqcn, level, addPrefix(message), null, exception);}}private static class LocationIgnorantKafkaLogger extends AbstractKafkaLogger {private final Logger logger;LocationIgnorantKafkaLogger(String logPrefix, Logger logger) {super(logPrefix);this.logger = logger;}@Overridepublic String getName() {return logger.getName();}@Overridepublic void info(String message) {logger.info(addPrefix(message));}}}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/127435.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Unix/Linux C语言 获取控制台窗口尺寸

在Unix/Linux控制台编程&#xff0c;为了能输出好看一些&#xff0c;需要知道窗口宽度&#xff0c;当然使用支持很宽的窗口的终端也是个办法&#xff0c;但是实在没有很宽的终端怎么办呢&#xff0c;还是要从程序上想办法的。 判断控制台窗口宽度需要两个函数&#xff1a; isa…

Java Soce

1.Server and client server 我们经常需要关闭一些实例&#xff0c;比如server&#xff0c;所以我们使用这个接口&#xff0c;来实现自动关闭。 我们可以这样写&#xff0c;手动关闭server public class Server {public static void main(String args[]){try {ServerSocket…

面向对象设计——装饰模式

装饰模式&#xff08;Decorator Pattern&#xff09;是一种结构型设计模式&#xff0c;它允许你动态地为对象添加额外的功能&#xff0c;而不需要修改其源代码。这种模式属于设计模式中的包装模式&#xff0c;它通过将对象包装在装饰器类中来实现。 装饰模式的核心思想是以透明…

panabit日志审计singleuser_action.php任意用户添加漏洞复现

文章目录 panabit日志审计singleuser_action.php任意用户添加漏洞复现0x01 前言0x02 漏洞描述0x03 影响版本0x04 漏洞环境0x05 漏洞复现1.访问漏洞环境2.构造POC3.复现 panabit日志审计singleuser_action.php任意用户添加漏洞复现 0x01 前言 免责声明&#xff1a;请勿利用文章…

flutter之bloc使用详解

flutter中一切皆为Widget&#xff0c;因此在我们开发中&#xff0c;往往业务和UI逻辑写在一起&#xff0c;这样不利于代码维护&#xff0c;因此状态管理框架久诞生了&#xff0c;这篇就开始讲一讲Bloc。 对于Bloc库有两个&#xff0c;如下图&#xff1a; flutter_bloc其实是对…

Ubuntu连不上WiFi 或者虽然能连上校园网,但是浏览器打不开登录页面

写在前面 自己的电脑环境&#xff1a; Ubuntu20.04 一、问题描述 自己的 Ubuntu 遇到连接不上 除校园网之外的其他WiFi, 或者 虽然能连上校园网&#xff0c;但是浏览器打不开登录页面的问题。 二、解决方法 出现这种问题的原因可能是 之前开过VPN, 导致系统的网络设置出现…

网络编程中关于UDP套接字的一些知识点

关于UDP的介绍&#xff1a; UDP&#xff08;User Datagram Protocol&#xff0c;用户数据报协议&#xff09;是一种面向无连接的、不可靠的传输协议&#xff0c;它在网络编程中也起着重要的作用。 1. 低延迟&#xff1a;相比于TCP&#xff0c;UDP没有建立连接和拥塞控制的开销…

系统架构设计师历年真题案例知识点汇总

常见的软件质量属性有多种&#xff0c;例如性能&#xff08;Performance)、可用性&#xff08;Availability)、可靠性&#xff08;Reliability)、健壮性&#xff08;Robustness)、安全性&#xff08;Security)、可修改性&#xff08;Modification)、可变性(Changeability)、易用…

MATLAB算法实战应用案例精讲-【图像处理】姿态估计

目录 前言 算法原理 姿态估计 2D姿态估计 3D姿态估计 3D形态估计 应用案例

1-性能分析-android-systrace

1-性能分析-android-systrace 一:Systrace简介二: Systrace 简单使用1> Systrace.py 介绍1. Systrace.py -h2. 常用参数2> 查看TAG三:systrace html 线程状态查看1> 线程状态-绿色 : 运行中(Running)2> 线程状态-蓝色 : 可运行(Runnable)3> 线程状态-白色…

CSS3网页布局基础

CSS布局始于第2个版本&#xff0c;CSS 2.1把布局分为3种模型&#xff1a;常规流、浮动、绝对定位。CSS 3推出更多布局方案&#xff1a;多列布局、弹性盒、模板层、网格定位、网格层、浮动盒等。本章重点介绍CSS 2.1标准的3种布局模型&#xff0c;它们获得所有浏览器的全面、一致…

基于深度学习的自动驾驶汽车语义分割与场景标注算法研究。

自动驾驶汽车是当前研究的热点领域之一&#xff0c;其中基于深度学习的语义分割与场景标注算法在自动驾驶汽车的视觉感知中具有重要作用。本文将围绕自动驾驶汽车的语义分割与场景标注算法展开研究。 一、研究背景 随着人工智能技术的不断发展&#xff0c;自动驾驶汽车逐渐成…

Golang 编译原理

简介 Golang&#xff08;Go语言&#xff09;是一种开源的编程语言&#xff0c;由Google开发并于2009年首次发布。它具备高效、可靠的特性&#xff0c;被广泛应用于云计算、分布式系统、网络服务等领域。Golang的编译原理是理解和掌握这门语言的重要基础之一。本文将介绍Golang…

回归预测 | Matlab实现POA-CNN-SVM鹈鹕算法优化卷积神经网络-支持向量机多变量回归预测

Matlab实现POA-CNN-SVM鹈鹕算法优化卷积神经网络-支持向量机多变量回归预测 目录 Matlab实现POA-CNN-SVM鹈鹕算法优化卷积神经网络-支持向量机多变量回归预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.POS-CNN-SVM鹈鹕算法优化卷积神经网络-支持向量机的多变量回归…

将有序数组转换为二叉搜索树

做这一题的前提是要搞懂一些概念&#xff0c;比如什么是高度平衡的二叉树&#xff1f;什么又是搜索树&#xff1f; 二叉搜索树&#xff08;Binary Search Tree&#xff09; 它或者是一棵空树&#xff0c;或者是具有下列性质的二叉树&#xff1a; 若它的左子树不空&#xff0c;则…

CV计算机视觉每日开源代码Paper with code速览-2023.10.31

精华置顶 墙裂推荐&#xff01;小白如何1个月系统学习CV核心知识&#xff1a;链接 点击CV计算机视觉&#xff0c;关注更多CV干货 论文已打包&#xff0c;点击进入—>下载界面 点击加入—>CV计算机视觉交流群 1.【基础网络架构】&#xff08;NeurIPS2023&#xff09;Fa…

EVM6678L 开发教程: IBL-TFTP 引导 elf 文件

目录 EVM6678L 开发教程: IBL-TFTP 引导 elf 文件安装 Tftpd64测试工程测试说明 EVM6678L 开发教程: IBL-TFTP 引导 elf 文件 参考: "C:\ti\mcsdk_2_01_02_06\tools\boot_loader\examples\i2c\tftp\docs\README.txt" 此教程介绍如何在 EVM6678L 开发板上实现 IBL-…

uni-starter 使用常见问题

1. Invalid uni-id config file 没有找到uni-id文件导致 需要在uniCloud-aliyun/cloudfunctions/common/uni-config-center/uni-id/下新建 config.json 如果没有uni-id 就新建一个。 注意&#xff1a;config.json是一个标准json文件&#xff0c;不支持注释 uni-starter 按照…

树形结构数据展示及返回上一级

11月1日&#xff0c;又是搬砖的一天&#xff0c;让我们红尘作伴&#xff0c;活的潇潇洒洒。。。。。。 html <template><view class"content"><view><input class"sreachTool" v-model"toolValue"/><van-icon name…

进口跨境电商商城源码(海关179接口+海关报关+三单对碰)

海关179接口 现如今&#xff0c;跨境电商正在飞速发展&#xff0c;进口商品成为人们消费的热点。然而&#xff0c;进口商品的报关手续繁琐&#xff0c;而海关179接口的出现解决了这个问题。海关179接口是指与海关电子数据交换的商业接口&#xff0c;可以实现与海关进行数据对接…