logback之pattern详解以及源码分析

目录

(一)pattern关键字介绍

(二)源码分析


(一)pattern关键字介绍

%d或%date:表示日期,可配置格式化%d{yyyy-MM-dd HH:mm:ss}

%r或%relative:也是日期,不过显示的是相对时间,是相对于日志框架加载开始的时间,单位是ms。

%level或%le或%p:表示日志级别

%t或%thread:表示线程名

%lo或%logger或%c(小写c):表示logger的名称,即 LoggerFactory.getLogger传入的名称,如果传入的是class则为类全限定名。

%m或%msg或%message:表示日志消息

%C(大写C)或%class:日志写入调用所在类名

%M或%method:日志写入调用所在的方法

 %L或者%line:日志写入调用所在的行数

%F或者%file:日志写入调用所在的文件

%X或%mdc:mdc写入的信息,记得mdc设置后,需要在finally中删除,避免内存泄漏。

%ex或%exception或%throwable:异常信息

%rEx或%rootException:带根因的异常信息

%xEx或%xException或%xThrowable:显示继承的异常信息

显示的异常信息量:%rootException > %xException > %exception;如果pattern中没有指定异常显示,那么默认会在末尾添加异常显示,根据上下文中的packagingDataEnabled设置,如果为true,加的是%xException,false为%exception(默认为false); %nopex或%nopexception:表示不打印异常信息。%ex在有异常的情况下回自动添加,而如果配置了这个那么就不会输出异常,所以和%ex是互斥的

%cn或%contextName:日志上下文的名称

%caller:表示日志详细的调用链

%maker:显示日志的标记,marker主要用于日志的分类和过滤。打印日志时可以调带Marker参数的方法,实现Marker的传递。

%property: 显示配置,会先查日志上下文的配置,如果没有从系统配置中获取,如:%property{java.version}就是打印java版本

%n: 表示换行

以下面格式的pattern,我们打印一个带异常的日志,看看效果,也好对应:

%d{yyyy-MM-dd HH:mm:ss} -%r [%thread] %-5level %logger{50} %class.%method-%line %F [%X{logTrackId}]
    [%cn] [%caller] [%marker] %property{java.version} - %msg%n

使用下面代码进行比较全的日志打印,这样和日志对照起来,对关键字就有了更具体的理解:

public void doNothing() {
	String logTrackId = UUID.randomUUID().toString();
	MDC.put("logTrackId", logTrackId);
	try {
		RuntimeException causeException = new RuntimeException("this is a cause exception");
		Marker marker = MarkerFactory.getMarker("basicMarker");
		ImpossibleLogger.warn(marker, "some exception happen:", new RuntimeException("this is encosing exception", causeException));
	} finally {
		MDC.remove("logTrackId");
	}
}

(二)源码分析

日志是如何根据pattern进行转化的呢?

下面我们看一下源码:

首先是<pattern>标签对应的PatternLayout类:这里定义了很多默认的转换器类:

可以看到关键字d和date都匹配的DateConverter。所以我们在pattern中定义的每个关键字都会有一个对应的转换器,既然是转换器那么一定会有转换的方法:convert,可以看一下DateConverter的convert方法,看一下它是如何转换的:

很简单只是把ILoggingEvent中获取时间戳,然后使用cachingDateFormatter进行转换,cachingDateFormatter实际上只是包装了一个SimpleDateFormat对象,因为其是线程不安全的,所以使用cachingDateFormatter将其包了一层,调用format时使用synchronized进行同步保证线程安全。

对于一个日志的格式化转换就是循环调用每个转换器然后将转换的信息添加到字符串中

那么转换器又是怎么加载到PatternLayout中的呢?可以看PatternLayoutBase.start方法,这个方法在Context初始化创建Pattern后会被调用。

核心的步骤有3个:

(1)创建一个解析器Parser,然后调用parse对pattern字符串进行解析,解析出的结果是一个Node层级对象,里面包含了%d、%thread等都被解析成对应的Node

(2)getEffectiveConverterMap获取有效的Converter集合,它主要把默认的转换器和以及context中PATTERN_RULE_REGISTRY存储自定义的转换器获取过来。

所以我们实际上可以通过往context中的PATTERN_RULE_REGISTRY这个key对应的map中写入自己的转换器类实现。key是关键字,值是自定义的转换器类名,而实现方式实际上是在配置文件中configuration节点下定义conversionRule标签,这个标签会被解析成converter,可参见ConversionRuleAction这个类

但要注意的是,xml的解析是有顺序的,所以conversionRule自定义的转换器需要在使用此转换器对应关键字的pattern之前,否则pattern解析时从PATTERN_RULE_REGISTRY获取不到对应的转换器。

(3)调用p.compile:这里就是转换器实际实例化创建的地方

第一步解析的Node有3中类型,分别是Node.LITERAL(字面量)、Node.COMPOSITE_KEYWORD(组合关键字)、Node.SIMPLE_KEYWORD(简单关键字)。字面量就是对应我们的pattern中的一些常量,如空格、-等会直接输出到日志中,不做转换。而组合转换器是比较复杂的转换器,会有子转换器的嵌套。我们常用的大部分都是SIMPLE_KEYWORD。

主要到DynamicConverter都会调用setFormattingInfo以及setOptionList,这些两个可能会被转换器用到。

如我们经常在%d后会加日期格式化yyyy-MM-dd HH:mm:ss,这个就会放到OptionList中,通过,分隔,如日期还可以设置时区%d{yyyy-MM-dd HH:mm:ss,GMT-8:00},这是OptionList中就有两个值

而%-5level中的-5则会设置到FormattingInfo,FormattingInfo主要设置关键字转换的字符串的最小长度和最大长度,用.分隔,如:%-5,7level,这里5就是最小值,7是最大值,表示日志级别最小显示长度为5,最大长度为7。最小值和最大值都可以带符号,最小值的正负号表示字符串长度不够时补齐的方向,-号是右边补齐,所以level如果是warn,那么不够5个字符串,那么右边会补一个空格。而最大值正负号则字符串超出,需要截取掉超出最大值的字符串,表示截取的方向,-号是截取左边的。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

vLLM结构化输出(Guided Decoding)

简介 vLLM 的结构化输出特性是通过“引导式解码”&#xff08;Guided Decoding&#xff09;实现的&#xff0c;这一功能允许模型在生成文本时遵循特定的格式约束&#xff0c;例如 JSON 模式或正则表达式&#xff0c;从而确保生成的内容符合预期的结构化要求。 后端引擎 启动…

CM3/CM4时钟系统

CM3/4时钟系统 1. CM3时钟系统1.1 输入时钟源------------------A1.2 锁相环PLL------------------B1.3 系统时钟SYSCLK--------C/D/E/F/G 2. CM4时钟系统2.1 输入时钟源------------------A2.2 锁相环PLL------------------B2.3 系统时钟SYSCLK--------C/D/E2.4 时钟信号输出M…

RabbitMQ实现生产者消费者

一.启动MQ 注意管理员身份进入cmd才行,我这里是在本地安装的MQ,推荐使用虚拟机安装 二.思路 官方解释RabbitMQ结构: 自我理解RabbitMQ结构: 其实RabbitMQ的服务器就像邮局一样,我们的生产者和消费者对于这个服务器来说都是消费者,因为服务器都可以向两者发送消息 环境准备 …

MySQL--》如何在SQL中巧妙运用函数与约束,优化数据处理与验证?

目录 函数使用 字符串函数 数值函数 日期函数 流程函数 约束 外键约束 约束规则 函数使用 函数是指一段可以直接被另一段程序调用的程序或代码&#xff0c;在mysql当中有许多常见的内置函数&#xff0c;接下来开始对这些内置函数及其作用进行简单的讲解和使用&#xf…

OpenLinkSaas使用手册-待办事项和通知中心

在OpenLinkSaas工作台上&#xff0c;你可以查看待办事项和未读通知。 待办事项 目前待办事项支持: 个人待办项目待办:在项目中指派给你的任务/缺陷Git待办:在Git仓库中指标给你的Issue,目前只有在AtomGit和Gitee账号登录时才支持。 通知中心 通知中心支持Git通知和邮件通知两种…

【Unity】 HTFramework框架(五十八)【进阶篇】资源及代码热更新实战演示(Deployment + HybridCLR)

更新日期&#xff1a;2025年1月2日。 Github源码&#xff1a;[点我获取源码] 索引 资源及代码热更新实战演示运行演示Demo1.克隆项目工程2.更新子模块3.打开项目4.打开入口场景5.设置远端资源服务器地址6.导入HybridCLR7.初始化HybridCLR8.发布项目9.部署资源版本10.运行Exe11.…

路由基本配置实验

路由器用于实现不同类型网络之间的互联。 路由器转发ip分组的基础是路由表。 路由表中的路由项分为直连路由项、静态路由项和动态路由项。 通过配置路由器接口的ip地址和子网掩码自动生成直连路由项。 通过手工配置创建静态路由项。 热备份路由器协议允许将由多个路由器组…

CTFshow—远程命令执行

29-35 Web29 代码利用正则匹配过滤了flag&#xff0c;后面加了/i所以不区分大小写。 可以利用通配符绕过 匹配任何字符串&#xff0f;文本&#xff0c;包括空字符串&#xff1b;*代表任意字符&#xff08;0个或多个&#xff09; ls file * ? 匹配任何一个字符&#xff08;不…

idea 的 springboot项目spring-boot-devtools 自动编译 配置热部署

1&#xff0c;设置一 2&#xff0c;设置二 设置二&#xff08;旧版本&#xff09; CtrlShiftAlt/ 点击弹出框中Registry... 引入&#xff08;如果报错&#xff0c;换不同的版本&#xff09; <dependency><groupId>org.springframework.boot</groupId><a…

Github拉取项目报错解决

前言 昨天在拉取github上面的项目报错了&#xff0c;有好几个月没用github了&#xff0c;命令如下&#xff1a; git clone gitgithub.com:zhszstudy/git-test.git报错信息&#xff1a; ssh: connect to host github.com port 22: Connection timed out fatal: Could not rea…

TypeScript 常用类型

文章目录 1. 类型注解2. 原始类型3. 数组类型4. 联合类型5. 类型别名6. 函数类型7. 对象类型8. 接口类型8.1 接口声明8.2 接口继承 9. 元组类型10. 类型断言11. 字面量类型12. 枚举类型12.1 数字枚举12.2 字符串枚举 13. any 类型14. typeof 运算符 1. 类型注解 前言&#xff1…

ARM200~500部署

前提&#xff1a;数据库已经安装好&#xff0c;并且正常运行 1.修改hostname,将里面的AR-A 改为hzx vi /etc/hostname 2.重启网络服务 sudo systemctl restart NetworkManager 3.修改community-admin.service 文件&#xff0c;更改小区名称和IP&#xff0c;并将文件上传到/…

Linux buildroot和ubuntu的异同点

Buildroot 和 Ubuntu 都是 Linux 系统的操作环境,但它们的设计理念和使用场景有很大的不同。 一、定义与目标 Buildroot Buildroot 是一个用于生成嵌入式 Linux 系统的工具集,专注于交叉编译和构建嵌入式设备的最小 Linux 环境。它的目标是为嵌入式系统提供定制化和优化的…

从0开始的opencv之旅(1)cv::Mat的使用

目录 Mat 存储方法 创建一个指定像素方式的图像。 尽管我们完全可以把cv::Mat当作一个黑盒&#xff0c;但是笔者的建议是仍然要深入理解和学习cv::Mat自身的构造逻辑和存储原理&#xff0c;这样在查找问题&#xff0c;或者是遇到一些奇奇怪怪的图像显示问题的时候能够快速的想…

免登录游客卡密发放系统PHP网站源码

源码介绍&#xff1a; 这是一个简单易用的卡密验证系统&#xff0c;主要功能包括&#xff1a; 卡密管理和验证&#xff0c;多模板支持&#xff0c;响应式设计&#xff0c;验证码保护&#xff0c;防刷机制&#xff0c;简洁的用户界面&#xff0c; 支持自定义模板&#xff0c;移…

LeetCode - 初级算法 数组(旋转数组)

旋转数组 这篇文章讨论如何通过编程实现数组元素的旋转操作。 免责声明:本文来源于个人知识与公开资料,仅用于学术交流。 描述 给定一个整数数组 nums,将数组中的元素向右轮转 k 个位置,其中 k 是非负数。 示例: 输入: nums = [1,2,3,

BOC调制信号matlab性能仿真分析,对比功率谱,自相关性以及抗干扰性

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 (完整程序运行后无水印) 2.算法运行软件版本 matlab2022a 3.部分核心程序 &#xff08;完整版代码包含详细中文注释和操作步骤视频&#xff09…

【从零开始入门unity游戏开发之——C#篇41】C#迭代器(Iterator)——自定义类实现 foreach 操作

文章目录 前言一、什么是迭代器&#xff1f;二、标准迭代器的实现方法1、自定义一个类CustomList2、让CustomList继承IEnumerable接口3、再继承IEnumerator接口4、完善迭代器功能5、**foreach遍历的本质**&#xff1a;6、在Reset方法里把光标复原 三、用yield return语法糖实现…

WordPress新安装只安装主题后发现只有首页能打开,其他路由页面都是404,并且Elementor都打不开

找到wordpress安装路径的这个文件&#xff0c;有发现里面没有内容&#xff0c;添加下面内容保存&#xff0c;重启服务器即可 # BEGIN WordPress <IfModule mod_rewrite.c> RewriteEngine On RewriteBase / RewriteRule ^index\.php$ – [L] RewriteCond %{REQUEST_FILEN…

uniapp中使用ruoyiPlus中的加密使用(crypto-js)

package.json中添加 "crypto-js": "^4.2.0", "jsencrypt": "^3.3.2",但是vue2中使用 import CryptoJS from cryptojs; 这一步就会报错 参照 参照这里&#xff1a;vue2使用CryptoJS实现信息加解密 根目录下的js文档中新增一个AESwork.…