Log4j2详细介绍
一、log4j、logback、log4j2 介绍
- log4j 是 apache 实现的一个开源日志组件
- logback 同样是由 log4j 的作者设计完成的,拥有更好的特性,用来取代 log4j 的一个日志框架,是 slf4j 的原生实现
- Log4j2 是 log4j 1.x 和 logback 的改进版,据说采用了一些新技术(无锁异步、等等),使得日志的吞吐量、性能比 log4j 1.x 提高 10 倍,并解决了一些死锁的 bug,而且配置更加简单灵活
了解一下,使用 slf4j+log4j 和直接用 log4j 的区别
slf4j是对所有日志框架制定的一种规范、标准、接口,并不是一个框架的具体的实现,因为接口并不能独立使用,需要和具体的日志框架实现配合使用(如 log4j、logback),使用接口的好处是当项目需要更换日志框架的时候,只需要更换 jar 和配置,不需要更改相关 java 代码。
log4j、logback、log4j2 都是一种日志具体实现框架,所以既可以单独使用也可以结合 slf4j 一起搭配使用。
如下面的是直接使用 log4j。
import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger;</span><span style="color: rgba(0, 0, 255, 1)">public</span> <span style="color: rgba(0, 0, 255, 1)">class</span><span style="color: rgba(0, 0, 0, 1)"> TestLog4j { </span><span style="color: rgba(0, 128, 0, 1)">//</span><span style="color: rgba(0, 128, 0, 1)"> Logger和LogManager导入的是org.apache.logging包</span> <span style="color: rgba(0, 0, 255, 1)">private</span> <span style="color: rgba(0, 0, 255, 1)">static</span> <span style="color: rgba(0, 0, 255, 1)">final</span> Logger LOG = LogManager.getLogger(TestLog4j.<span style="color: rgba(0, 0, 255, 1)">class</span><span style="color: rgba(0, 0, 0, 1)">); }</span></pre>
二、log4j2 的使用介绍
1、导入需要使用的 jar 包 (slf4j+log4j2)
springboot 项目中需导入:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-log4j2</artifactId> </dependency>
绕坑:如项目中有导入 spring-boot-starter-web 依赖包记得去掉 spring 自带的日志依赖 spring-boot-starter-logging,如下:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> <exclusions> <exclusion> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-logging</artifactId> </exclusion> </exclusions> </dependency>
2、开始配置
springboot 方式:
application.properties 中添加配置 logging.config=classpath:log4j2_dev.xml, log4j2_dev.xml 是你创建的 log4j2 的配置文件名,放在 resources 下,如放在其他路径则对应修改
配置文件的格式:log2j 配置文件可以是 xml 格式的,也可以是 json 格式的,
配置文件的位置:log4j2 默认会在 classpath 目录下寻找 log4j2.xml、log4j.json、log4j.jsn 等名称的文件,如果都没有找到,则会按默认配置输出,也就是输出到控制台,也可以对配置文件自定义位置(需要在 web.xml 中配置),一般放置在 src/main/resources 根目录下即可
贴上 log4j2_dev.properties 的配置再来讲解
<?xml version="1.0" encoding="UTF-8"?> <!--设置 log4j2 的自身 log 级别为 warn--> <!--日志级别以及优先级排序: OFF > FATAL > ERROR > WARN > INFO > DEBUG > TRACE > ALL --> <!--Configuration 后面的 status,这个用于设置 log4j2 自身内部的信息输出,可以不设置, 当设置成 trace 时,你会看到 log4j2 内部各种详细输出--> <!--monitorInterval:Log4j 能够自动检测修改配置 文件和重新配置本身,设置间隔秒数--> <configuration status="warn" monitorInterval="30"> <!--先定义所有的 appender--> <appenders> <!--这个输出控制台的配置--> <console name="Console" target="SYSTEM_OUT"> <!--输出日志的格式--> <PatternLayout pattern="[%d{HH🇲🇲ss:SSS}] [%p] - %l - %m%n"/> </console> <!--文件会打印出所有信息,这个 log 每次运行程序会自动清空,由 append 属性决定,这个也挺有用的,适合临时测试用--> <File name="log" fileName="log/test.log" append="false"> <PatternLayout pattern="%d{HH🇲🇲ss.SSS} %-5level %class{36} %L %M - %msg%xEx%n"/> </File> <!-- 这个会打印出所有的 info 及以下级别的信息,每次大小超过 size, 则这 size 大小的日志会自动存入按年份 - 月份建立的文件夹下面并进行压缩,作为存档--> <RollingFile name="RollingFileInfo" fileName="${sys:user.home}/logs/hpaasvc/info.log" filePattern="${sys:user.home}/logs/hpaasvc/$${date:yyyy-MM}/info-%d{yyyy-MM-dd}-%i.log"> <!-- level 日志级别 onMatch 符合日志级别的处理情况 onMisMatch 不符合日志级别的处理情况 ACCEPT 接受打印 DENY 拒绝打印 NEUTRAL 不做处理 即忽略此 filter 看看下面有没有能够处理的规则 以下两条翻译: 在 warn 及以上的级别都不打印,其他的不做处理 > WARN\ERROR\FATAL 被过滤删除 只剩 TRACE\DEBUG\INFO 在 info 及以上的都打印,其他的不打印 > RACE\DEBUG 被过滤删除 只剩下 info 被打印 所以要注意 Filter 的执行顺序 --> <Filters> <!--控制台只输出 level 及以上级别的信息(onMatch),其他的直接拒绝(onMismatch)--> <ThresholdFilter level="INFO" onMatch="ACCEPT" onMismatch="DENY"/> <ThresholdFilter level="WARN" onMatch="DENY" onMismatch="NEUTRAL"/> </Filters> <PatternLayout pattern="[%d{HH🇲🇲ss:SSS}] [%p] - %l - %m%n"/> <Policies> <TimeBasedTriggeringPolicy/> <SizeBasedTriggeringPolicy size="100 MB"/> </Policies> </RollingFile><span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">RollingFile </span><span style="color: rgba(255, 0, 0, 1)">name</span><span style="color: rgba(0, 0, 255, 1)">="RollingFileWarn"</span><span style="color: rgba(255, 0, 0, 1)"> fileName</span><span style="color: rgba(0, 0, 255, 1)">="${sys:user.home}/logs/hpaasvc/warn.log"</span><span style="color: rgba(255, 0, 0, 1)"> filePattern</span><span style="color: rgba(0, 0, 255, 1)">="${sys:user.home}/logs/hpaasvc/$${date:yyyy-MM}/warn-%d{yyyy-MM-dd}-%i.log"</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">Filters</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">ThresholdFilter </span><span style="color: rgba(255, 0, 0, 1)">level</span><span style="color: rgba(0, 0, 255, 1)">="WARN"</span><span style="color: rgba(255, 0, 0, 1)"> onMatch</span><span style="color: rgba(0, 0, 255, 1)">="ACCEPT"</span><span style="color: rgba(255, 0, 0, 1)"> onMismatch</span><span style="color: rgba(0, 0, 255, 1)">="DENY"</span><span style="color: rgba(0, 0, 255, 1)">/></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">ThresholdFilter </span><span style="color: rgba(255, 0, 0, 1)">level</span><span style="color: rgba(0, 0, 255, 1)">="ERROR"</span><span style="color: rgba(255, 0, 0, 1)"> onMatch</span><span style="color: rgba(0, 0, 255, 1)">="DENY"</span><span style="color: rgba(255, 0, 0, 1)"> onMismatch</span><span style="color: rgba(0, 0, 255, 1)">="NEUTRAL"</span><span style="color: rgba(0, 0, 255, 1)">/></span> <span style="color: rgba(0, 0, 255, 1)"></</span><span style="color: rgba(128, 0, 0, 1)">Filters</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">PatternLayout </span><span style="color: rgba(255, 0, 0, 1)">pattern</span><span style="color: rgba(0, 0, 255, 1)">="[%d{HH🇲🇲ss:SSS}] [%p] - %l - %m%n"</span><span style="color: rgba(0, 0, 255, 1)">/></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">Policies</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">TimeBasedTriggeringPolicy</span><span style="color: rgba(0, 0, 255, 1)">/></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">SizeBasedTriggeringPolicy </span><span style="color: rgba(255, 0, 0, 1)">size</span><span style="color: rgba(0, 0, 255, 1)">="100 MB"</span><span style="color: rgba(0, 0, 255, 1)">/></span> <span style="color: rgba(0, 0, 255, 1)"></</span><span style="color: rgba(128, 0, 0, 1)">Policies</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 128, 0, 1)"><!--</span><span style="color: rgba(0, 128, 0, 1)"> DefaultRolloverStrategy属性如不设置,则默认为最多同一文件夹下7个文件,这里设置了20 </span><span style="color: rgba(0, 128, 0, 1)">--></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">DefaultRolloverStrategy </span><span style="color: rgba(255, 0, 0, 1)">max</span><span style="color: rgba(0, 0, 255, 1)">="20"</span><span style="color: rgba(0, 0, 255, 1)">/></span> <span style="color: rgba(0, 0, 255, 1)"></</span><span style="color: rgba(128, 0, 0, 1)">RollingFile</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">RollingFile </span><span style="color: rgba(255, 0, 0, 1)">name</span><span style="color: rgba(0, 0, 255, 1)">="RollingFileError"</span><span style="color: rgba(255, 0, 0, 1)"> fileName</span><span style="color: rgba(0, 0, 255, 1)">="${sys:user.home}/logs/hpaasvc/error.log"</span><span style="color: rgba(255, 0, 0, 1)"> filePattern</span><span style="color: rgba(0, 0, 255, 1)">="${sys:user.home}/logs/hpaasvc/$${date:yyyy-MM}/error-%d{yyyy-MM-dd}-%i.log"</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">ThresholdFilter </span><span style="color: rgba(255, 0, 0, 1)">level</span><span style="color: rgba(0, 0, 255, 1)">="ERROR"</span><span style="color: rgba(0, 0, 255, 1)">/></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">PatternLayout </span><span style="color: rgba(255, 0, 0, 1)">pattern</span><span style="color: rgba(0, 0, 255, 1)">="[%d{HH🇲🇲ss:SSS}] [%p] - %l - %m%n"</span><span style="color: rgba(0, 0, 255, 1)">/></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">Policies</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">TimeBasedTriggeringPolicy</span><span style="color: rgba(0, 0, 255, 1)">/></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">SizeBasedTriggeringPolicy </span><span style="color: rgba(255, 0, 0, 1)">size</span><span style="color: rgba(0, 0, 255, 1)">="100 MB"</span><span style="color: rgba(0, 0, 255, 1)">/></span> <span style="color: rgba(0, 0, 255, 1)"></</span><span style="color: rgba(128, 0, 0, 1)">Policies</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"></</span><span style="color: rgba(128, 0, 0, 1)">RollingFile</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"></</span><span style="color: rgba(128, 0, 0, 1)">appenders</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 128, 0, 1)"><!--</span><span style="color: rgba(0, 128, 0, 1)">然后定义logger,只有定义了logger并引入的appender,appender才会生效</span><span style="color: rgba(0, 128, 0, 1)">--></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">loggers</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 128, 0, 1)"><!--</span><span style="color: rgba(0, 128, 0, 1)"> additivity="false"表示在该logger中输出的日志不会再延伸到父层logger </span><span style="color: rgba(0, 128, 0, 1)">--></span> <span style="color: rgba(0, 128, 0, 1)"><!--</span><span style="color: rgba(0, 128, 0, 1)">过滤掉spring和hibernate的一些无用的debug信息</span><span style="color: rgba(0, 128, 0, 1)">--></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">logger </span><span style="color: rgba(255, 0, 0, 1)">name</span><span style="color: rgba(0, 0, 255, 1)">="org.springframework"</span><span style="color: rgba(255, 0, 0, 1)"> level</span><span style="color: rgba(0, 0, 255, 1)">="INFO"</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"></</span><span style="color: rgba(128, 0, 0, 1)">logger</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">logger </span><span style="color: rgba(255, 0, 0, 1)">name</span><span style="color: rgba(0, 0, 255, 1)">="org.mybatis"</span><span style="color: rgba(255, 0, 0, 1)"> level</span><span style="color: rgba(0, 0, 255, 1)">="INFO"</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"></</span><span style="color: rgba(128, 0, 0, 1)">logger</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">root </span><span style="color: rgba(255, 0, 0, 1)">level</span><span style="color: rgba(0, 0, 255, 1)">="all"</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">appender-ref </span><span style="color: rgba(255, 0, 0, 1)">ref</span><span style="color: rgba(0, 0, 255, 1)">="Console"</span><span style="color: rgba(0, 0, 255, 1)">/></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">appender-ref </span><span style="color: rgba(255, 0, 0, 1)">ref</span><span style="color: rgba(0, 0, 255, 1)">="RollingFileInfo"</span><span style="color: rgba(0, 0, 255, 1)">/></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">appender-ref </span><span style="color: rgba(255, 0, 0, 1)">ref</span><span style="color: rgba(0, 0, 255, 1)">="RollingFileWarn"</span><span style="color: rgba(0, 0, 255, 1)">/></span> <span style="color: rgba(0, 0, 255, 1)"><</span><span style="color: rgba(128, 0, 0, 1)">appender-ref </span><span style="color: rgba(255, 0, 0, 1)">ref</span><span style="color: rgba(0, 0, 255, 1)">="RollingFileError"</span><span style="color: rgba(0, 0, 255, 1)">/></span> <span style="color: rgba(0, 0, 255, 1)"></</span><span style="color: rgba(128, 0, 0, 1)">root</span><span style="color: rgba(0, 0, 255, 1)">></span> <span style="color: rgba(0, 0, 255, 1)"></</span><span style="color: rgba(128, 0, 0, 1)">loggers</span><span style="color: rgba(0, 0, 255, 1)">></span>
</configuration>
异步日志:
<?xml version="1.0" encoding="UTF-8"?> <Configuration status="WARN"> <properties> <property name = "LOG_HOME">D:/logs</property> <property name = "INFO_NAME">info</property> <property name = "DEBUG_NAME">debug</property> <property name = "ERROR_NAME">error</property> <property name = "LOG_NAME">log</property> </properties> <Appenders> <Console name="Console" target="SYSTEM_OUT"> <PatternLayout pattern="%d{HH🇲🇲ss.SSS} [%t] %-5level %logger{36} - %msg%n" /> </Console> <RollingRandomAccessFile name="MyFile" fileName="${LOG_HOME}/${LOG_NAME}.log" filePattern="${LOG_HOME}/$${date:yyyy-MM}/${LOG_NAME}-%d{yyyy-MM-dd}-%i.log"> <PatternLayout pattern="%d{yyyy-MM-dd HH🇲🇲ss.SSS} [%t] %-5level %logger{36} - %msg%n" /> <Policies> <TimeBasedTriggeringPolicy interval="1" /> <SizeBasedTriggeringPolicy size="10 MB" /> </Policies> <DefaultRolloverStrategy max="20" /> </RollingRandomAccessFile> <Async name="myAsync"> <AppenderRef ref="MyFile" /> </Async> </Appenders> <Loggers> <Logger name="myLog" level="trace" additivity = "false"> <AppenderRef ref="myAsync"/> </Logger> <Root level="error"> <AppenderRef ref="Console" /> </Root> </Loggers> </Configuration>
log4j2.xml 文件的配置大致如下:
- Configuration
- properties
- Appenders
- Console
- PatternLayout
- File
- RollingRandomAccessFile
- Async
- Console
- Loggers
- Logger
- Root
- AppenderRef
-
Configuration:为根节点,有 status 和 monitorInterval 等多个属性
- status 的值有 “trace”, “debug”, “info”, “warn”, “error” and “fatal”,用于控制 log4j2 日志框架本身的日志级别,如果将 stratus 设置为较低的级别就会看到很多关于 log4j2 本身的日志,如加载 log4j2 配置文件的路径等信息
- monitorInterval,含义是每隔多少秒重新读取配置文件,可以不重启应用的情况下修改配置
-
Appenders:输出源,用于定义日志输出的地方
log4j2 支持的输出源有很多,有控制台 Console、文件 File、RollingRandomAccessFile、MongoDB、Flume 等-
Console:控制台输出源是将日志打印到控制台上,开发的时候一般都会配置,以便调试
-
File:文件输出源,用于将日志写入到指定的文件,需要配置输入到哪个位置(例如:D:/logs/mylog.log)
-
RollingRandomAccessFile: 该输出源也是写入到文件,不同的是比 File 更加强大,可以指定当文件达到一定大小(如 20MB)时,另起一个文件继续写入日志,另起一个文件就涉及到新文件的名字命名规则,因此需要配置文件命名规则
这种方式更加实用,因为你不可能一直往一个文件中写,如果一直写,文件过大,打开就会卡死,也不便于查找日志。- fileName 指定当前日志文件的位置和文件名称
- filePattern 指定当发生 Rolling 时,文件的转移和重命名规则
- SizeBasedTriggeringPolicy 指定当文件体积大于 size 指定的值时,触发 Rolling
- DefaultRolloverStrategy 指定最多保存的文件个数
- TimeBasedTriggeringPolicy 这个配置需要和 filePattern 结合使用,注意 filePattern 中配置的文件重命名规则是 ${FILE_NAME}-%d{yyyy-MM-dd HH-mm}-%i,最小的时间粒度是 mm,即分钟
- TimeBasedTriggeringPolicy 指定的 size 是 1,结合起来就是每 1 分钟生成一个新文件。如果改成 %d{yyyy-MM-dd HH},最小粒度为小时,则每一个小时生成一个文件
-
NoSql:MongoDb, 输出到 MongDb 数据库中
-
Flume:输出到 Apache Flume(Flume 是 Cloudera 提供的一个高可用的,高可靠的,分布式的海量日志采集、聚合和传输的系统,Flume 支持在日志系统中定制各类数据发送方,用于收集数据;同时,Flume 提供对数据进行简单处理,并写到各种数据接受方(可定制)的能力。)
-
Async:异步,需要通过 AppenderRef 来指定要对哪种输出源进行异步(一般用于配置 RollingRandomAccessFile)
PatternLayout:控制台或文件输出源(Console、File、RollingRandomAccessFile)都必须包含一个 PatternLayout 节点,用于指定输出文件的格式(如 日志输出的时间 文件 方法 行数 等格式),例如 pattern=”%d{HH🇲🇲ss.SSS} [%t] %-5level %logger{36} - %msg%n”
-
%d{HH🇲🇲ss.SSS} 表示输出到毫秒的时间
-
%t 输出当前线程名称
-
%-5level 输出日志级别,-5 表示左对齐并且固定输出5 个字符,如果不足在右边补0
-
%logger 输出 logger 名称,因为 Root Logger 没有名称,所以没有输出
-
%msg 日志文本
-
%n 换行
-
-
其他常用的占位符有:
-
%F 输出所在的类文件名,如 Log4j2Test.java
-
%L 输出行号
-
%M 输出所在方法名
-
%l 输出语句所在的行数, 包括类名、方法名、文件名、行数
-
-
Loggers:日志器
日志器分根日志器 Root 和自定义日志器,当根据日志名字获取不到指定的日志器时就使用 Root 作为默认的日志器,自定义时需要指定每个 Logger 的名称 name(对于命名可以以包名作为日志的名字,不同的包配置不同的级别等),日志级别 level,相加性 additivity(是否继承下面配置的日志器), 对于一般的日志器(如 Console、File、RollingRandomAccessFile)一般需要配置一个或多个输出源 AppenderRef;每个 logger 可以指定一个 level(TRACE, DEBUG, INFO, WARN, ERROR, ALL or OFF),不指定时 level 默认为 ERROR
additivity 指定是否同时输出 log 到父类的 appender,缺省为 true。
-
<Logger name="rollingRandomAccessFileLogger" level="trace" additivity="true">
-
<AppenderRef ref="RollingRandomAccessFile" />
-
</Logger>
- properties: 属性
使用来定义常量,以便在其他配置的时候引用,该配置是可选的,例如定义日志的存放位置
D:/logs
三、特性
关于 log4j2 的新特性
丢数据这种情况少,可以用来做审计功能。而且自身内部报的 exception 会被发现,但是 logback 和 log4j 不会。
log4j2 使用了 disruptor 技术,在多线程环境下,性能高于 logback 等 10 倍以上。
(garbage free)之前的版本会产生非常多的临时对象,会造成 GC 频繁,log4j2 则在这方面上做了优化,减少产生临时对象。尽可能少的 GC
利用插件系统,使得扩展新的 appender,filter,layout 等变得容易,log4j 不可以扩展 插件????
因为插件系统的简单性,所以在配置的时候,可以不用具体指定所要处理的类型。class
可以自定义 level
Java 8 lambda support for lazy logging
Support for Message objects
对 filter 的功能支持的更强大
系统日志 (Syslog) 协议 supports both TCP and UDP
利用 jdk1.5 并发的特性,减少了死锁的发生。
Socket LogEvent SerializedLayout
支持 kafka queue
四、问题:
1. 日志不按小时滚动
原来的配置如下:
<RollingRandomAccessFile name="RollingFile" fileName="${backupFilePatch}${fileName}" filePattern="${backupFilePatch}${fileName}.%d{yyyy-MM-dd}.%i" immediateFlush="false"> <PatternLayout pattern="%d{HH🇲🇲ss.SSS} %p %c{1}[%L]-%m%n"/> <Policies> <TimeBasedTriggeringPolicy interval="1" modulate="true"/> </Policies> <DefaultRolloverStrategy max="120"/> </RollingRandomAccessFile>
根据官网的介绍,我理解成了 interval=1 的时候固定按小时滚动,其实 interval 的单位是根据 filePattern 来的,即 filePattern 配置中的最小单位,所以在 yyyy-MM-dd 这样的配置中,interval 的单位是 day,而 %i 则是 interval 的序号,日志是按天滚动的。所以将配置改为 ${backupFilePatch}${fileName}.%d{yyyy-MM-dd.HH} 就能按小时滚动了。
2. 每小时滚动后,日志不写入新文件,而是写入旧的归档文件。
这个问题更诡异。tomcat 启动后,日志是写入 log 文件的,但是整点日志滚动之后,log 归档为 log- 日期,产生一个新的 log 文件,但日志还是打入了归档之后的文件 log- 日期。放置了一天之后,发现更诡异的事情,日志总是写入上一个归档文件。例如,7 点的时候,产生一个归档文件 log- 日期.6,7 点到 8 点的日志都打入了文件 6,log 文件始终是空的,而 8 点之后,log 归档为 log- 日期 -7,之后的日志都打入文件 7,新产生的 log 文件还是为空。
该问题遍寻不着解法。在某一次整点时,正好刷新了一下,发现 log 文件有新内容,再刷新又成为上面那种现象了。由此联想到,有可能不只是 log4j2 在对这个文件进行操作。
排查发现,改写 log4j2.xml 之后,原来的 log4j.properties 还留在项目里。由于 maven 层层叠叠的依赖,有几个包引用的是 log4j1.x 版本,在运行的时候,始终有一个包用的是 log4j1.x 在打日志,因此加载的是 log4j.properties 的配置。log4j.properties 和 log4j2.xml 的配置完全相同,写入的也是同一个日志文件。整点的时候,这两个类都对 log 文件进行归档操作,猜想是 log4j2 先执行归档,写入新 log 文件,而这个文件又被 log4j1 归档,因此后面 log4j2 的日志都写入了上一个归档文件。
将 log4j.properties 中的日志改为其他名称,问题解决。