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)">&lt;</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)">&gt;</span>
        <span style="color: rgba(0, 0, 255, 1)">&lt;</span><span style="color: rgba(128, 0, 0, 1)">Filters</span><span style="color: rgba(0, 0, 255, 1)">&gt;</span>
            <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">/&gt;</span>
            <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">/&gt;</span>
        <span style="color: rgba(0, 0, 255, 1)">&lt;/</span><span style="color: rgba(128, 0, 0, 1)">Filters</span><span style="color: rgba(0, 0, 255, 1)">&gt;</span>
        <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">/&gt;</span>
        <span style="color: rgba(0, 0, 255, 1)">&lt;</span><span style="color: rgba(128, 0, 0, 1)">Policies</span><span style="color: rgba(0, 0, 255, 1)">&gt;</span>
            <span style="color: rgba(0, 0, 255, 1)">&lt;</span><span style="color: rgba(128, 0, 0, 1)">TimeBasedTriggeringPolicy</span><span style="color: rgba(0, 0, 255, 1)">/&gt;</span>
            <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">/&gt;</span>
        <span style="color: rgba(0, 0, 255, 1)">&lt;/</span><span style="color: rgba(128, 0, 0, 1)">Policies</span><span style="color: rgba(0, 0, 255, 1)">&gt;</span>
        <span style="color: rgba(0, 128, 0, 1)">&lt;!--</span><span style="color: rgba(0, 128, 0, 1)"> DefaultRolloverStrategy属性如不设置,则默认为最多同一文件夹下7个文件,这里设置了20 </span><span style="color: rgba(0, 128, 0, 1)">--&gt;</span>
        <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">/&gt;</span>
    <span style="color: rgba(0, 0, 255, 1)">&lt;/</span><span style="color: rgba(128, 0, 0, 1)">RollingFile</span><span style="color: rgba(0, 0, 255, 1)">&gt;</span>

    <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">&gt;</span>
        <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">/&gt;</span>
        <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">/&gt;</span>
        <span style="color: rgba(0, 0, 255, 1)">&lt;</span><span style="color: rgba(128, 0, 0, 1)">Policies</span><span style="color: rgba(0, 0, 255, 1)">&gt;</span>
            <span style="color: rgba(0, 0, 255, 1)">&lt;</span><span style="color: rgba(128, 0, 0, 1)">TimeBasedTriggeringPolicy</span><span style="color: rgba(0, 0, 255, 1)">/&gt;</span>
            <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">/&gt;</span>
        <span style="color: rgba(0, 0, 255, 1)">&lt;/</span><span style="color: rgba(128, 0, 0, 1)">Policies</span><span style="color: rgba(0, 0, 255, 1)">&gt;</span>
    <span style="color: rgba(0, 0, 255, 1)">&lt;/</span><span style="color: rgba(128, 0, 0, 1)">RollingFile</span><span style="color: rgba(0, 0, 255, 1)">&gt;</span>

<span style="color: rgba(0, 0, 255, 1)">&lt;/</span><span style="color: rgba(128, 0, 0, 1)">appenders</span><span style="color: rgba(0, 0, 255, 1)">&gt;</span>
<span style="color: rgba(0, 128, 0, 1)">&lt;!--</span><span style="color: rgba(0, 128, 0, 1)">然后定义logger,只有定义了logger并引入的appender,appender才会生效</span><span style="color: rgba(0, 128, 0, 1)">--&gt;</span>
<span style="color: rgba(0, 0, 255, 1)">&lt;</span><span style="color: rgba(128, 0, 0, 1)">loggers</span><span style="color: rgba(0, 0, 255, 1)">&gt;</span>
   <span style="color: rgba(0, 128, 0, 1)">&lt;!--</span><span style="color: rgba(0, 128, 0, 1)"> additivity="false"表示在该logger中输出的日志不会再延伸到父层logger </span><span style="color: rgba(0, 128, 0, 1)">--&gt;</span>
    <span style="color: rgba(0, 128, 0, 1)">&lt;!--</span><span style="color: rgba(0, 128, 0, 1)">过滤掉spring和hibernate的一些无用的debug信息</span><span style="color: rgba(0, 128, 0, 1)">--&gt;</span>
    <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">&gt;</span>
    <span style="color: rgba(0, 0, 255, 1)">&lt;/</span><span style="color: rgba(128, 0, 0, 1)">logger</span><span style="color: rgba(0, 0, 255, 1)">&gt;</span>
    <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">&gt;</span>
    <span style="color: rgba(0, 0, 255, 1)">&lt;/</span><span style="color: rgba(128, 0, 0, 1)">logger</span><span style="color: rgba(0, 0, 255, 1)">&gt;</span>
    <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">&gt;</span>
        <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">/&gt;</span>
        <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">/&gt;</span>
        <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">/&gt;</span>
        <span style="color: rgba(0, 0, 255, 1)">&lt;</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)">/&gt;</span>
    <span style="color: rgba(0, 0, 255, 1)">&lt;/</span><span style="color: rgba(128, 0, 0, 1)">root</span><span style="color: rgba(0, 0, 255, 1)">&gt;</span>
<span style="color: rgba(0, 0, 255, 1)">&lt;/</span><span style="color: rgba(128, 0, 0, 1)">loggers</span><span style="color: rgba(0, 0, 255, 1)">&gt;</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
    • 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”

    1. %d{HH🇲🇲ss.SSS} 表示输出到毫秒的时间
    2. %t 输出当前线程名称
    3. %-5level 输出日志级别,-5 表示左对齐并且固定输出5 个字符,如果不足在右边补0
    4. %logger 输出 logger 名称,因为 Root Logger 没有名称,所以没有输出
    5. %msg 日志文本
    6. %n 换行
    7.  
    8. 其他常用的占位符有:
    9. %F 输出所在的类文件名,如 Log4j2Test.java
    10. %L 输出行号
    11. %M 输出所在方法名
    12. %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。

  1. <Logger name="rollingRandomAccessFileLogger" level="trace" additivity="true">
  2. <AppenderRef ref="RollingRandomAccessFile" />
  3. </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 中的日志改为其他名称,问题解决。