Log4j2详解

article/2025/9/18 16:45:18

1.背景

Log4j 2中记录日志的方式有同步日志和异步日志两种方式,其中异步日志又可分为使用AsyncAppender和使用AsyncLogger两种方式。

2.Log4j2中的同步日志

所谓同步日志,即当输出日志时,必须等待日志输出语句执行完毕后,才能执行后面的业务逻辑语句。

下面通过一个例子来了解Log4j2中的同步日志,并借此来探究整个日志输出过程。

log4j2.xml配置如下:

复制代码

<?xml version="1.0" encoding="UTF-8"?>
<Configuration status="debug" name="MyApp" packages=""><!--全局Filter--><ThresholdFilter level="ALL"/><Appenders><Console name="Console" target="SYSTEM_OUT"><PatternLayout pattern="%d{HH:mm:ss.SSS} [%t] %-5level %logger{36} - %msg%n" /></Console><RollingFile name="RollingFile" fileName="logs/app.log"filePattern="logs/app-%d{yyyy-MM-dd HH}.log"><!--Appender的Filter--><ThresholdFilter level="info" onMatch="ACCEPT" onMismatch="DENY"/><PatternLayout><Pattern>%d %p %c{1.} [%t] %m%n</Pattern></PatternLayout><Policies><SizeBasedTriggeringPolicy size="500MB"/></Policies></RollingFile></Appenders><Loggers><Logger name="com.meituan.Main" level="trace" additivity="false"><!--Logger的Filter--><ThresholdFilter level="debug"/><appender-ref ref="RollingFile"/></Logger><Root level="debug"><AppenderRef ref="Console"/></Root></Loggers>
</Configuration>

复制代码

java代码如下:

复制代码

import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;public class Main {public static void main(String args[]) {Logger logger = LogManager.getLogger(Main.class);Person person = new Person("Li", "lei");logger.info("hello, {}", person);}private static class Person {private String firstName;private String lastName;public Person(String firstName, String lastName) {this.firstName = firstName;this.lastName = lastName;}public String toString() {return "Person[" + firstName + "," + lastName + "]";}}
}

复制代码

使用以上的配置,当我们运行程序后,以下log将被添加到logs/app.log中。

2017-09-13 19:41:00,889 INFO c.m.Main [main] hello, Person[Li,lei]

logger.info运行时到底发生了什么?日志信息是如何输出到app.log中的?

Log4j2中日志输出的详细过程如下:

1.首先使用全局Filter对日志事件进行过滤。

Log4j2中的日志Level分为8个级别,优先级从高到低依次为OFF、FATAL、ERROR、WARN、INFO、DEBUG、TRACE、 ALL。

全局Filter的Level为ALL,表示允许输出所有级别的日志。logger.info()请求输出INFO级别的日志,通过。

2.使用Logger的Level对日志事件进行过滤。

Logger的Level为TRACE,表示允许输出TRACE级别及以上级别的日志。logger.info()请求输出INFO级别的日志,通过。

3.生成日志输出内容Message。

使用占位符的方式输出日志,输出语句为logger.info("increase {} from {} to {}", arg1, arg2, arg3)的形式,最终输出时{}占位符处的内容将用arg1,arg2,arg3的字符串填充。

log4j2用Object[]保存参数信息,在这一阶段会将Object[]转换为String[],生成含有输出模式串"increase {} from {} to {}"和参数数组String[]的Message,为后续日志格式化输出做准备。

4.生成LogEvent。

LogEvent中含有loggerName(日志的输出者),level(日志级别),timeMillis(日志的输出时间),message(日志输出内容),threadName(线程名称)等信息。

在上述程序中,生成的LogEvent的属性值为loggerName=com.meituan.Main,Level=INFO,timeMillis=1505659461759,message为步骤3中创建的Message,threadNama=main。

5.使用Logger配置的Filter对日志事件进行过滤。

Logger配置的Filter的Level为DEBUG,表示允许输出DEBUG及以上级别的日志。logger.info()请求输出INFO级别的日志,通过。

6.使用Logger对应的Appender配置的Filter对日志事件进行过滤。

Appender配置的Filter配置的INFO级别日志onMatch=ACCEPT,表示允许输出INFO级别的日志。logger.info()请求输出INFO级别的日志,通过。

7.判断是否需要触发rollover。

此步骤不是日志输出的必须步骤,如配置的Appender为无需进行rollover的Appender,则无此步骤。

因为使用RollingFileAppender,且配置了基于文件大小的rollover触发策略,在此阶段会判断是否需要触发rollover。判断方式为当前的文件大小是否达到了指定的size,如果达到了,触发rollover操作。关于Log4j2中的RollingFileAppender的rollover,可参见Log4j2中RollingFile的文件滚动更新机制。

8.PatternLayout对LogEvent进行格式化,生成可输出的字符串。

上述log4j2.xml文件中配置的Pattern及各个参数的意义如下:

<Pattern>%d %p %c{1.} [%t] %m%n</Pattern>

参数

意义

%d
日期格式,默认形式为2012-11-02 14:34:02,781
%p
日志级别
%c{1.}
%c表示Logger名字,{1.}表示精确度。若Logger名字为org.apache.commons.Foo,则输出o.a.c.Foo。
%t
处理LogEvent的线程的名字
%m
日志内容
%n
行分隔符。"\n"或"\r\n"。

在此步骤,PatternLayout将根据Pattern的模式,利用各种Converter对LogEvent的相关信息进行转换,最终拼接成可输出的日志字符串。

如DatePatternConverter对LogEvent的日志输出时间进行格式化转换;LevelPatternConverter对LogEvent的日志级别信息进行格式化转换;LoggerPatternConverter对LogEvent的Logger的名字进行格式化转换;MessagePatternConverter对LogEvent的日志输出内容进行格式化转换等。

经各种Converter转换后,LogEvent的信息被格式化为指定格式的字符串。

9.使用OutputStream,将日志输出到文件。

将日志字符串序列化为字节数组,使用字节流OutoutStream将日志输出到文件中。如果配置了immediateFlush为true,打开app.log就可观察到输出的日志了。

3.Log4j2中的异步日志

使用log4j2的同步日志进行日志输出,日志输出语句与程序的业务逻辑语句将在同一个线程运行,如上面的例子,打印的日志中显示的线程名称为main,与业务逻辑语句在同一个线程中(此句有误,LogEvent中的线程名称不是输出LogEvent的线程,是生成LogEvent的线程,输出LogEvent的线程和生成LogEvent可能不是一个线程!)

而使用异步日志进行输出时,日志输出语句与业务逻辑语句并不是在同一个线程中运行,而是有专门的线程用于进行日志输出操作,处理业务逻辑的主线程不用等待即可执行后续业务逻辑。

Log4j2中的异步日志实现方式有AsyncAppender和AsyncLogger两种。

其中,AsyncAppender采用了ArrayBlockingQueue来保存需要异步输出的日志事件;AsyncLogger则使用了Disruptor框架来实现高吞吐。

3.1 AsyncAppender

复制代码

<?xml version="1.0" encoding="UTF-8"?>
<Configuration status="warn"><Appenders><RollingFile name="MyFile" fileName="logs/app.log"><PatternLayout><Pattern>%d %p %c{1.} [%t] %m%n</Pattern></PatternLayout><SizeBasedTriggeringPolicy size="500MB"/></RollingFile><Async name="Async"><AppenderRef ref="MyFile"/></Async></Appenders><Loggers><Root level="error"><AppenderRef ref="Async"/></Root></Loggers>
</Configuration>

复制代码

上面就是一个使用AsyncAppender的典型配置,配置AsyncAppender后,日志事件写入文件的操作将在单独的线程中执行。

AsyncAppender的常用参数

参数名

类型

说明

nameStringAsync Appender的名字。
AppenderRefString

异步调用的Appender的名字,可以配置多个。

blockingboolean

默认为true。如果为true,appender将一直等待直到queue中有空闲;如果为false,当队列满的时候,日志事件将被丢弃。(如果配置了error appender,要丢弃的日志事件将由error appender处理)

bufferSizeinteger

队列中可存储的日志事件的最大数量,默认为128。(源码中为128,Log4j2官网为1024,官网信息有误)

关于AsyncAppender的其他参数,可参考Log4j2对AsyncAppender的详细介绍。

每个Async Appender,内部维护了一个ArrayBlockingQueue,并将创建一个线程用于输出日志事件,如果配置了多个AppenderRef,将分别使用对应的Appender进行日志输出。

3.2 AsyncLogger

Log4j2中的AsyncLogger的内部使用了Disruptor框架。

Disruptor简介

Disruptor是英国外汇交易公司LMAX开发的一个高性能队列,基于Disruptor开发的系统单线程能支撑每秒600万订单。

目前,包括Apache Strom、Log4j2在内的很多知名项目都应用了Disruptor来获取高性能。

Disruptor框架内部核心数据结构为RingBuffer,其为无锁环形队列。

 

单线程每秒能够处理600万订单,Disruptor为什么这么快?

a.lock-free-使用了CAS来实现线程安全

ArrayBlockingQueue使用锁实现并发控制,当get或put时,当前访问线程将上锁,当多生产者、多消费者的大量并发情形下,由于锁竞争、线程切换等,会有性能损失。

Disruptor通过CAS实现多生产者、多消费者对RingBuffer的并发访问。CAS相当于乐观锁,其性能优于Lock的性能。

b.使用缓存行填充解决伪共享问题

计算机体系结构中,内存的访问速度远远低于CPU的运行速度,在内存和CPU之间,加入Cache,CPU首先访问Cache中的数据,CaChe未命中,才访问内存中的数据。

伪共享:Cache是以缓存行(cache line)为单位存储的,当多个线程修改互相独立的变量时,如果这些变量共享同一个缓存行,就会无意中影响彼此的性能。

 

关于伪共享的深度分析,可参考《伪共享,并发编程的性能杀手》这篇文章。

AsyncLogger

Log4j2异步日志如何进行日志输出,我们同样从一个例子出发来探究Log4j2的异步日志。

log4j2.xml配置如下:

复制代码

<?xml version="1.0" encoding="UTF-8"?>
<Configuration status="debug" name="MyApp" packages=""><Appenders><Console name="Console" target="SYSTEM_OUT"><PatternLayout pattern="%d{HH:mm:ss.SSS} [%t] %-5level %logger{36} - %msg%n" /></Console><RollingFile name="RollingFile" fileName="logs/app.log"filePattern="logs/app-%d{yyyy-MM-dd HH}.log"><PatternLayout><Pattern>%d %p %c{1.} [%t] %m%n</Pattern></PatternLayout><Policies><SizeBasedTriggeringPolicy size="500MB"/></Policies></RollingFile><RollingFile name="RollingFile2" fileName="logs/app2.log"filePattern="logs/app2-%d{yyyy-MM-dd HH}.log"><PatternLayout><Pattern>%d %p %c{1.} [%t] %m%n</Pattern></PatternLayout><Policies><SizeBasedTriggeringPolicy size="500MB"/></Policies></RollingFile></Appenders><Loggers><AsyncLogger name="com.meituan.Main" level="trace" additivity="false"><appender-ref ref="RollingFile"/></AsyncLogger><AsyncLogger name="RollingFile2" level="trace" additivity="false"><appender-ref ref="RollingFile2"/></AsyncLogger><Root level="debug"><AppenderRef ref="Console"/><AppenderRef ref="RollingFile"/></Root></Loggers>
</Configuration>

复制代码

java代码如下:

复制代码

public class Main {public static void main(String args[]) {Logger logger = LogManager.getLogger(Main.class);Logger logger2 = LogManager.getLogger("RollingFile2");Person person = new Person("Li", "lei");logger.info("hello, {}", person);logger2.info("good bye, {}", person);
}

复制代码

上述log4j2.xml中配置了两个AsyncLogger,名字分别为com.meituan.Main和RollingFile2。

并且,在main方法中分别使用两个logger来输出两条日志。

 

在加载log4j2.xml的启动阶段,如果检测到配置了AsyncRoot或AsyncLogger,将启动一个disruptor实例。

上述程序中,main线程作为生产者,EventProcessor线程作为消费者。

生产者生产消息

当运行到类似于logger.info、logger.debug的输出语句时,将生成的LogEvent放入RingBuffer中。

消费者消费消息

如果RingBuffer中有LogEvent需要处理,EventProcessor线程从RingBuffer中取出LogEvent,调用Logger相关联的Appender输出LogEvent(具体输出过程与同步过程相同,同样需要过滤器过滤、PatternLayout格式化等步骤)。

如果RingBuffer中没有LogEvent需要处理,EventProcessor线程将处于等待阻塞状态(默认策略)。

需要注意的是,虽然在log4j2.xml中配置了多个AsyncLogger,但是并不是每个AsyncLogger对应着一个处理线程,而是仅仅有一个EventProcessor线程进行日志的异步处理。

4.总结

 

日志输出方式

sync同步打印日志,日志输出与业务逻辑在同一线程内,当日志输出完毕,才能进行后续业务逻辑操作
Async Appender异步打印日志,内部采用ArrayBlockingQueue,对每个AsyncAppender创建一个线程用于处理日志输出。
Async Logger异步打印日志,采用了高性能并发框架Disruptor,创建一个线程用于处理日志输出。

http://chatgpt.dhexx.cn/article/DypoaiY4.shtml

相关文章

log4j2 的使用【超详细图文】

log4j2 的使用 Apache Log4j2 是对Log4j 的升级版本&#xff0c;参考了logback 的一些优秀的设计&#xff0c;并且修复了一些问题&#xff0c;因此带来了一些重大的提升&#xff0c;主要有&#xff1a; 异常处理&#xff0c;在logback中&#xff0c;Appender中的异常不会被应…

Log4j2 简介

介绍 Log4j2是Log4j的升级版&#xff0c;与之前的版本Log4j 1.x相比、有重大的改进&#xff0c;修正了Logback固有的架构问题的同事&#xff0c;改进了许多Logback所具有的功能。 特性 一、API 分离 Log4j2将API与实现分离开来。开发人员现在可以很清楚的知道能够使用哪些没有兼…

数据分析方法08用户画像

用户画像 1、 概念 用户画像是指用户信息标签化&#xff0c;通过收集用户多维度的信息数据&#xff08;如人口统计属性&#xff0c;社会属性&#xff0c;行为偏好&#xff0c;消费习惯等&#xff09;&#xff0c;对其进行统计&#xff0c;分析&#xff0c;从而抽象出用户信息…

用户画像及其应用案例分享

一、用户画像基础概念 用户画像&#xff0c;即用户信息标签化&#xff0c;通过收集用户的社会属性、消费习惯、偏好特征等各个维度的数据&#xff0c;对用户特征属性进行刻画&#xff0c;并对这些特征进行分析、统计&#xff0c;挖掘潜在价值信息&#xff0c;从而抽象出用户的…

大数据分析--用户画像

从去年还在北京工作的时候就开始不断的听说&#xff0c;用户画像、用户画像&#xff0c;而且&#xff0c;有个项目&#xff0c;项目领导美其名曰&#xff1a;用户画像&#xff0c;这么长时间&#xff0c;也没有真正的思考过什么叫用户画像。前几天看到一篇关于这方面的文章&…

【数据分析】京东平台用户画像分析

1.用户信息标签 用户信息的收集包括用户基础要素、用户场景、行为偏好、心理个性、交际等多方面。 可以用静态标签、动态标签两大类来划分。 静态数据:用户相对稳定的信息&#xff0c;主要包括人口属性、商业属性等方面数据&#xff1b;这类信息果企业有真实信息则无需过多建…

用户画像解析

用户画像是大数据行业言必及之的时髦概念。现在我们运营谈及用户画像&#xff0c;它也是和精准营销、精细化运营直接钩挂的。这篇文章主要讲产品和运营角度的用户画像。 一、什么是用户画像 用户画像一点也不神秘&#xff0c;它是根据用户在互联网留下的种种数据&#xff0c;主…

用户画像分析总结

一、 什么是用户画像 用户画像是指根据用户的属性、用户偏好、生活习惯、用户行为等信息而抽象出来的标签化用户模型。通俗说就是给用户打标签&#xff0c;而标签是通过对用户信息分析而来的高度精炼的特征标识。通过打标签可以利用一些高度概括、容易理解的特征来描述用户&…

用户画像详解来了

导读&#xff1a;今天主要和大家分享明略对于用户画像的认识&#xff0c;包括如何理解用户画像&#xff0c;为什么需要用户用画像&#xff0c;以及明略在建设和应用用户画像过程中总结的方法论&#xff0c;让大家了解如何构建好的用户画像。 01 什么是用户画像 首先来看用户画像…

Python的import

Python里面的import语句用于导入一个已经写好的包或者模块 使得自己的程序里面可以使用导入的模块里面的函数、变量等 import name 导入numpy模块&#xff0c;里面有一个shape函数&#xff0c;要使用这个函数&#xff0c;numpy.shape()即可 但是不加numpy就不行&#xff0c;…

python-import this

优美胜于丑陋&#xff08;python 以编写优美的代码为目标&#xff09; 明了胜于晦涩&#xff08;优美的代码应该是明了的&#xff0c;命名规范&#xff0c;风格相似&#xff09; 简洁胜于复杂&#xff08;优美的代码应当是简洁的&#xff0c;不要有复杂的内部实现&#xff09…

import re

元字符 px* 可以匹配 p px px... px 可以匹配 px px... 但是不能匹配到p默认的匹配都是贪婪匹配 贪婪匹配就是一次匹配尽可能多的内容比如正则是 aoe 要匹配的字符串是 aoeeeeee 它可以匹配到aoeeeeee 但是如果正则是 aoe? 的话 它只能匹配到aoe ?放在 * {n,m}是取消贪婪匹…

1.连接服务器

1 使用服务器 参考文章&#xff1a;https://blog.csdn.net/longgeaisisi/article/details/78680180 双11通过团购入手阿里云服务器会得到一个提货券&#xff0c;同时收到短信提醒&#xff0c;跟着短信链接&#xff0c;进入阿里云控制台使用提货券。然后&#xff0c;会出现选项…

windows下连接服务器

一、windows下连接服务器 xhsell 是一个windows的软件 是用来连接远程服务器的 使用的是ssh协议 xftp 一个windows的软件 用来给服务器 上传或者下载文件的一个工具 ftp协议 shell环境 是运行命令的环境 shell脚本 用于给运维人员 来去做批量处理的 一个语言shell语言编写的一个…

linux之连接服务器

1. 背景 通常连接服务器使用ssh usernamehostname的方式&#xff0c;其中uername是用户名&#xff0c;一般为root等。hostname为ip地址。但是ip地址通常不容易记住&#xff0c;因此会将ip地址与一个名称相对应&#xff0c;然后直接使用名字来登陆服务器。 2. ip与name的映射 …

远程服务器怎么连接

利用windows自带远程桌面连接服务器&#xff0c;海星星 日付服务器 碉堡了 工具/原料 远程桌面 IP地址账号 方法/步骤 1 本商品直接使用WINDOWS自带的远程桌面连接就可以进行使用。 点开始——点运行&#xff08;或者直接用窗口键[win]R&#xff09;————在运行对话框输入m…

连接服务器

连接是什么意思 创建套接字之后&#xff0c;应用程序&#xff08;浏览器&#xff09;就会调用connect&#xff0c;随后协议栈会将本地的套接字与服务器的套接字进行连接。话说&#xff0c;以太网的网线都是一直连接的状态&#xff0c;我们并不需要来回插拔网线&#xff0c;那么…

创建链接服务器

GPS平台、网站建设、软件开发、系统运维&#xff0c;找森大网络科技&#xff01; https://cnsendnet.taobao.com 来自森大科技官方博客 http://www.cnsendblog.com/index.php/?p1862 给SQL Server创建连接服务器 一、 可视化界面操作 1.点击“新建连接服务器” 2.“常规”选…

如何连接云服务器

** - 本篇文章是&#xff1a;教跟我一样一开始不会连接远程云服务器的小白们 ** 首先你已经有一个公网ip&#xff08;在阿里云/腾讯云/等等许多地方都可以租到&#xff09; 注&#xff1a;作者本人申请了一个试用的华为云2008Windows64bit的服务器 以下教你如何一步步连接…

连接到服务器(1)

问题一&#xff08;未找到或无法访问服务器&#xff09; 标题: 连接到服务器 ------------------------------ 无法连接到 TOM-PC。 ------------------------------ 其他信息: 在与 SQL Server 建立连接时出现与网络相关的或特定于实例的错误。未找到或无法访问服务器。请验证…