log4j升级到log4j2,并配合druid插入数据库

2022元旦前报的全球Log4j日志严重漏洞问题,第一次报漏洞的时候我这边版本不在影响范围内,又过了两个月通知我:公司要求全部更新到最新版log4j2。

当时正烦躁在别的项目,于是就安排别的人员去升级。。。

最近手里不忙,发现半年多过去了,md,没给我升级。。。

靠人不如靠自己,还得自己来,顺便看看最新的log4j2有什么变化。

项目旧版本用的log4j.1.2.16,够老吧,截止今天(2022.08.27)最新版本是log4j.2.18.0。

总体来说log4j2比log4j更快更方便更安全(但是也爆出来天大的漏洞)。。。

下面分别贴出升级过程中log4j和log4j2版本的修改配置。

<!-- pom.xml配置 -->
<!-- 原log4j -->
<dependency>
    <groupId>log4j</groupId>
    <artifactId>log4j</artifactId>
    <version>1.2.16</version>
</dependency>

<!-- 新log4j2 -->
<dependency>
    <groupId>org.apache.logging.log4j</groupId>
    <artifactId>log4j-api</artifactId>
     <version>2.18.0</version>
</dependency>
<dependency>
    <groupId>org.apache.logging.log4j</groupId>
    <artifactId>log4j-core</artifactId>
    <version>2.18.0</version>
</dependency>

有的文章介绍说还要引入org.slf4j,但这个我项目中并没用到,不加也正常使用,难道这个是兼容低版本log4j用的? 

<!-- web.xml关于原log4j的配置 -->
    <!-- 加载log4j配置文件 -->
    <context-param>
        <param-name>log4jConfigLocation</param-name>
        <param-value>/WEB-INF/classes/log4j.properties</param-value>
    </context-param>
    <!--log4jRefreshInterval为100000表示 开一条watchdog线程每10秒扫描一下配置文件的变化; -->
    <context-param>
        <param-name>log4jRefreshInterval</param-name>
        <param-value>100000</param-value>
    </context-param>
    <!-- 配置log4j文件路径独一无二 -->
    <context-param>
        <param-name>webAppRootKey</param-name>
        <param-value>adminapi.root</param-value>
    </context-param>
    <!-- 1. 动态的改变记录级别和策略,不需要重启Web应用; 2. 把log文件定在 /WEB-INF/logs/ 而不需要写绝对路径。 -->
    <listener>
        <listener-class>org.springframework.web.util.Log4jConfigListener</listener-class>
    </listener>

下面网上查的资料还要加 org.apache.logging.log4j.web.Log4jServletContextListener 的配置,但我这边加上后报错找不到该类,去掉后正常。

<!-- web.xml关于新log4j2的配置 -->
    <context-param>
        <param-name>log4jConfiguration</param-name>
        <param-value>classpath:log4j2.xml</param-value>
    </context-param>

<!-- 网上的资料说还要加下面这段配置,但我的加上报错,去掉后正常 -->
    <listener>
           <listener-class>org.apache.logging.log4j.web.Log4jServletContextListener</listener-class>
    </listener>

如上web.xml中之前log4j对于日志的详细配置log4jConfiguration文件写在了/WEB-INF/classes/log4j.properties文件,现在log4j2推荐写在log4j2.xml中,项目中是放在resource文件夹下。内容如下:

#原log4j配置properties #####################
#OFF、FATAL、ERROR、WARN、INFO、DEBUG、ALL

#未定义的logger,默认只输出ERROR级别以上的到stdout
log4j.rootLogger = ERROR, stdout

#各个java项目需要根据各自的命名空间,修改 #adminapi# 这个部分
log4j.logger.com.adminapi.mapper    = WARN, mapper
log4j.logger.ex.debug                        = DEBUG, ex
log4j.logger.mail.debug                        = DEBUG, mail
log4j.logger.db.debug                       = DEBUG, mysql

log4j.appender.mapper                = org.apache.log4j.DailyRollingFileAppender
log4j.appender.mapper.File            = /data/logs/pc.admin/api.mapper.log
log4j.appender.mapper.DatePattern    = '.'yyyy-MM-dd
log4j.appender.mapper.Append        = true
log4j.appender.mapper.layout        = org.apache.log4j.PatternLayout
log4j.appender.mapper.layout.ConversionPattern = %m%n

log4j.appender.mail                = org.apache.log4j.DailyRollingFileAppender
log4j.appender.mail.File        = /data/logs/pc.admin/api.mail.log
log4j.appender.mail.DatePattern    = '.'yyyy-MM-dd
log4j.appender.mail.Append        = true
log4j.appender.mail.layout        = org.apache.log4j.PatternLayout
log4j.appender.mail.layout.ConversionPattern = %-d{yyyy-MM-dd HH\:mm\:ss} [%p] %C{1}.%M:%L %m %n

log4j.appender.ex                = org.apache.log4j.DailyRollingFileAppender
log4j.appender.ex.File            = /data/logs/pc.admin/api.ex.log
log4j.appender.ex.DatePattern    = '.'yyyy-MM-dd
log4j.appender.ex.Append        = true
log4j.appender.ex.layout        = org.apache.log4j.PatternLayout
log4j.appender.ex.layout.ConversionPattern = %-d{yyyy-MM-dd HH\:mm\:ss} [%p] %C{1}.%M:%L %m %n

log4j.appender.mysql            = org.apache.log4j.jdbc.JDBCAppender
log4j.appender.mysql.driver        = com.mysql.jdbc.Driver
log4j.appender.mysql.sql        = INSERT INTO yx_log4j(level,class,function,line,message) VALUES('%p','%C','%M',%L,'%m');
log4j.appender.mysql.layout        = org.apache.log4j.PatternLayout
log4j.appender.mysql.URL        = ${jdbc.url}
log4j.appender.mysql.user        = ${jdbc.username}
log4j.appender.mysql.password    = ${jdbc.password}
#输出到控制台
log4j.appender.stdout            = org.apache.log4j.ConsoleAppender
log4j.appender.stdout.layout    = org.apache.log4j.PatternLayout

新log4j2.xml配置如下

<!-- 新log4j2配置log4j2.xml -->
<?xml version="1.0" encoding="UTF-8"?>
<!--日志级别以及优先级排序: OFF > FATAL > ERROR > WARN > INFO > DEBUG > TRACE > ALL -->
<!--Configuration后面的status,这个用于设置log4j2自身内部的信息输出,可以不设置,当设置成trace时,你会看到log4j2内部各种详细输出-->
<!--monitorInterval:Log4j能够自动检测修改配置 文件和重新配置本身,设置间隔秒数-->
<configuration status="WARN" monitorInterval="1800">
 
    <Properties>
        <!-- ==============================================公共配置============================================== -->
        <!-- 设置日志文件的目录名称 -->
        <property name="logFileName">adminLog4jLog</property>
 
        <!-- 日志默认存放的位置,可以设置为项目根路径下,也可指定绝对路径 -->
        <!-- 存放路径一:通用路径,window平台 -->
        <!-- <property name="basePath">d:/logs/${logFileName}</property> -->
        <!-- 存放路径二:web工程专用,java项目没有这个变量,需要删掉,否则会报异常,这里把日志放在web项目的根目录下 -->
        <!-- <property name="basePath">${web:rootDir}/${logFileName}</property> -->
        <!-- 存放路径三:web工程专用,java项目没有这个变量,需要删掉,否则会报异常,这里把日志放在tocmat的logs目录下 -->
        <property name="basePath">/data/logs/${logFileName}</property>
 
        <!-- 控制台默认输出格式,"%-5level":日志级别,"%l":输出完整的错误位置,是小写的L,因为有行号显示,所以影响日志输出的性能 -->
        <property name="console_log_pattern">%d{yyyy-MM-dd HH:mm:ss.SSS} [%-5level] %l - %m%n</property>
        <!-- 日志文件默认输出格式,不带行号输出(行号显示会影响日志输出性能);%C:大写,类名;%M:方法名;%m:错误信息;%n:换行 -->
        <!-- <property name="log_pattern">%d{yyyy-MM-dd HH:mm:ss.SSS} [%-5level] %C.%M - %m%n</property> -->
        <!-- 日志文件默认输出格式,另类带行号输出(对日志输出性能未知);%C:大写,类名;%M:方法名;%L:行号;%m:错误信息;%n:换行 -->
        <property name="log_pattern">%d{yyyy-MM-dd HH:mm:ss.SSS} [%-5level] %C.%M[%L line] - %m%n</property>
 
        <!-- 日志默认切割的最小单位 -->
        <property name="every_file_size">20MB</property>
        <!-- 日志默认输出级别 -->
        <property name="output_log_level">DEBUG</property>
 
        <!-- ===========================================所有级别日志配置=========================================== -->
        <!-- 日志默认存放路径(所有级别日志) -->
        <property name="rolling_fileName">${basePath}/all.log</property>
        <!-- 日志默认压缩路径,将超过指定文件大小的日志,自动存入按"年月"建立的文件夹下面并进行压缩,作为存档 -->
        <property name="rolling_filePattern">${basePath}/%d{yyyy-MM}/all-%d{yyyy-MM-dd-HH}-%i.log.gz</property>
        <!-- 日志默认同类型日志,同一文件夹下可以存放的数量,不设置此属性则默认为7个,filePattern最后要带%i才会生效 -->
        <property name="rolling_max">500</property>
        <!-- 日志默认同类型日志,多久生成一个新的日志文件,这个配置需要和filePattern结合使用;
                        如果设置为1,filePattern是%d{yyyy-MM-dd}到天的格式,则间隔一天生成一个文件
                        如果设置为12,filePattern是%d{yyyy-MM-dd-HH}到小时的格式,则间隔12小时生成一个文件 -->
        <property name="rolling_timeInterval">12</property>
        <!-- 日志默认同类型日志,是否对封存时间进行调制,若为true,则封存时间将以0点为边界进行调整,
                如:现在是早上3am,interval是4,那么第一次滚动是在4am,接着是8am,12am...而不是7am -->
        <property name="rolling_timeModulate">true</property>
 
        <!-- ============================================Info级别日志============================================ -->
        <!-- Info日志默认存放路径(Info级别日志) -->
        <property name="info_fileName">${basePath}/info.log</property>
        <!-- Info日志默认压缩路径,将超过指定文件大小的日志,自动存入按"年月"建立的文件夹下面并进行压缩,作为存档 -->
        <property name="info_filePattern">${basePath}/%d{yyyy-MM}/info-%d{yyyy-MM-dd}-%i.log.gz</property>
        <!-- Info日志默认同一文件夹下可以存放的数量,不设置此属性则默认为7个 -->
        <property name="info_max">100</property>
        <!-- 日志默认同类型日志,多久生成一个新的日志文件,这个配置需要和filePattern结合使用;
                        如果设置为1,filePattern是%d{yyyy-MM-dd}到天的格式,则间隔一天生成一个文件
                        如果设置为12,filePattern是%d{yyyy-MM-dd-HH}到小时的格式,则间隔12小时生成一个文件 -->
        <property name="info_timeInterval">1</property>
        <!-- 日志默认同类型日志,是否对封存时间进行调制,若为true,则封存时间将以0点为边界进行调整,
                    如:现在是早上3am,interval是4,那么第一次滚动是在4am,接着是8am,12am...而不是7am -->
        <property name="info_timeModulate">true</property>
 
        <!-- ============================================Warn级别日志============================================ -->
        <!-- Warn日志默认存放路径(Warn级别日志) -->
        <property name="warn_fileName">${basePath}/warn.log</property>
        <!-- Warn日志默认压缩路径,将超过指定文件大小的日志,自动存入按"年月"建立的文件夹下面并进行压缩,作为存档 -->
        <property name="warn_filePattern">${basePath}/%d{yyyy-MM}/warn-%d{yyyy-MM-dd}-%i.log.gz</property>
        <!-- Warn日志默认同一文件夹下可以存放的数量,不设置此属性则默认为7个 -->
        <property name="warn_max">100</property>
        <!-- 日志默认同类型日志,多久生成一个新的日志文件,这个配置需要和filePattern结合使用;
                        如果设置为1,filePattern是%d{yyyy-MM-dd}到天的格式,则间隔一天生成一个文件
                        如果设置为12,filePattern是%d{yyyy-MM-dd-HH}到小时的格式,则间隔12小时生成一个文件 -->
        <property name="warn_timeInterval">1</property>
        <!-- 日志默认同类型日志,是否对封存时间进行调制,若为true,则封存时间将以0点为边界进行调整,
                    如:现在是早上3am,interval是4,那么第一次滚动是在4am,接着是8am,12am...而不是7am -->
        <property name="warn_timeModulate">true</property>
 
        <!-- ============================================Error级别日志============================================ -->
        <!-- Error日志默认存放路径(Error级别日志) -->
        <property name="error_fileName">${basePath}/error.log</property>
        <!-- Error日志默认压缩路径,将超过指定文件大小的日志,自动存入按"年月"建立的文件夹下面并进行压缩,作为存档 -->
        <property name="error_filePattern">${basePath}/%d{yyyy-MM}/error-%d{yyyy-MM-dd}-%i.log.gz</property>
        <!-- Error日志默认同一文件夹下可以存放的数量,不设置此属性则默认为7个 -->
        <property name="error_max">100</property>
        <!-- 日志默认同类型日志,多久生成一个新的日志文件,这个配置需要和filePattern结合使用;
                        如果设置为1,filePattern是%d{yyyy-MM-dd}到天的格式,则间隔一天生成一个文件
                        如果设置为12,filePattern是%d{yyyy-MM-dd-HH}到小时的格式,则间隔12小时生成一个文件 -->
        <property name="error_timeInterval">1</property>
        <!-- 日志默认同类型日志,是否对封存时间进行调制,若为true,则封存时间将以0点为边界进行调整,
                    如:现在是早上3am,interval是4,那么第一次滚动是在4am,接着是8am,12am...而不是7am -->
        <property name="error_timeModulate">true</property>
 
        <!-- ============================================控制台显示控制============================================ -->
        <!-- 控制台显示的日志最低级别 -->
        <property name="console_print_level">DEBUG</property>
    </Properties>
 
    <!--定义appender -->
    <appenders>
        <!-- =======================================用来定义输出到控制台的配置======================================= -->
        <Console name="Console" target="SYSTEM_OUT">
            <!-- 设置控制台只输出level及以上级别的信息(onMatch),其他的直接拒绝(onMismatch)-->
            <ThresholdFilter level="${console_print_level}" onMatch="ACCEPT" onMismatch="DENY"/>
            <!-- 设置输出格式,不设置默认为:%m%n -->
            <PatternLayout pattern="${console_log_pattern}"/>
        </Console>
 
        <!-- ================================打印root中指定的level级别以上的日志到文件================================ -->
        <RollingFile name="RollingFile" fileName="${rolling_fileName}" filePattern="${rolling_filePattern}">
            <PatternLayout pattern="${log_pattern}"/>
            <Policies>
                <TimeBasedTriggeringPolicy interval="${rolling_timeInterval}" modulate="${rolling_timeModulate}"/>
                <SizeBasedTriggeringPolicy size="${every_file_size}"/>
            </Policies>
            <DefaultRolloverStrategy max="${rolling_max}" />
        </RollingFile>
 
        <!-- =======================================打印INFO级别的日志到文件======================================= -->
        <RollingFile name="InfoFile" fileName="${info_fileName}" filePattern="${info_filePattern}">
            <PatternLayout pattern="${log_pattern}"/>
            <Policies>
                <TimeBasedTriggeringPolicy interval="${info_timeInterval}" modulate="${info_timeModulate}"/>
                <SizeBasedTriggeringPolicy size="${every_file_size}"/>
            </Policies>
            <DefaultRolloverStrategy max="${info_max}" />
            <Filters>
                <ThresholdFilter level="WARN" onMatch="DENY" onMismatch="NEUTRAL"/>
                <ThresholdFilter level="INFO" onMatch="ACCEPT" onMismatch="DENY"/>
            </Filters>
        </RollingFile>
 
        <!-- =======================================打印WARN级别的日志到文件======================================= -->
        <RollingFile name="WarnFile" fileName="${warn_fileName}" filePattern="${warn_filePattern}">
            <PatternLayout pattern="${log_pattern}"/>
            <Policies>
                <TimeBasedTriggeringPolicy interval="${warn_timeInterval}" modulate="${warn_timeModulate}"/>
                <SizeBasedTriggeringPolicy size="${every_file_size}"/>
            </Policies>
            <DefaultRolloverStrategy max="${warn_max}" />
            <Filters>
                <ThresholdFilter level="ERROR" onMatch="DENY" onMismatch="NEUTRAL"/>
                <ThresholdFilter level="WARN" onMatch="ACCEPT" onMismatch="DENY"/>
            </Filters>
        </RollingFile>
 
        <!-- =======================================打印ERROR级别的日志到文件======================================= -->
        <RollingFile name="ErrorFile" fileName="${error_fileName}" filePattern="${error_filePattern}">
            <PatternLayout pattern="${log_pattern}"/>
            <Policies>
                <TimeBasedTriggeringPolicy interval="${error_timeInterval}" modulate="${error_timeModulate}"/>
                <SizeBasedTriggeringPolicy size="${every_file_size}"/>
            </Policies>
            <DefaultRolloverStrategy max="${error_max}" />
            <Filters>
                <ThresholdFilter level="FATAL" onMatch="DENY" onMismatch="NEUTRAL"/>
                <ThresholdFilter level="ERROR" onMatch="ACCEPT" onMismatch="DENY"/>
            </Filters>
        </RollingFile>
    </appenders>
 
    <!--定义logger,只有定义了logger并引入的appender,appender才会生效-->
    <loggers>
        <!-- 设置打印sql语句配置开始,以下两者配合使用,可以优化日志的输出信息,减少一些不必要信息的输出 -->
        <!-- 设置java.sql包下的日志只打印DEBUG及以上级别的日志,此设置可以支持sql语句的日志打印 -->
        <!--<logger name="java.sql" level="DEBUG" additivity="false">
            <appender-ref ref="Console"/>
        </logger>
        &lt;!&ndash; 设置org.mybatis.spring包下的日志只打印WARN及以上级别的日志 &ndash;&gt;
        <logger name="org.mybatis.spring" level="WARN" additivity="false">
            <appender-ref ref="Console"/>
        </logger>
        &lt;!&ndash; 设置org.springframework包下的日志只打印WARN及以上级别的日志 &ndash;&gt;
        <logger name="org.springframework" level="WARN" additivity="false">
            <appender-ref ref="Console"/>
        </logger>
        &lt;!&ndash; 设置com.qfx.workflow.service包下的日志只打印WARN及以上级别的日志 &ndash;&gt;
        <logger name="com.qfx.workflow.service" level="WARN" additivity="false">
            <appender-ref ref="Console"/>
        </logger>-->
 
        <logger name="net.sf.ehcache" level="DEBUG" additivity="false">
            <appender-ref ref="Console" />
        </logger>
        <logger name="org.springframework" level="INFO" additivity="false">
            <appender-ref ref="Console" />
        </logger>
 
        <logger name="org.hibernate" level="INFO" additivity="false">
            <appender-ref ref="Console" />
        </logger>
 
        <!-- log4jdbc-remix -->
        <logger name="jdbc.sqlonly" level="WARN" additivity="false">
            <appender-ref ref="Console" />
        </logger>
        <logger name="jdbc.sqltiming" level="INFO" additivity="false">
            <appender-ref ref="Console" />
        </logger>
        <logger name="jdbc.connection" level="INFO" additivity="false">
            <appender-ref ref="Console" />
        </logger>
        <logger name="jdbc.resultset" level="WARN" additivity="false">
            <appender-ref ref="Console" />
        </logger>
        <logger name="jdbc.resultsettable" level="WARN" additivity="false">
            <appender-ref ref="Console" />
        </logger>
        <logger name="jdbc.audit" level="WARN" additivity="false">
            <appender-ref ref="Console" />
        </logger>
 
        <logger name="java.sql.Connection" level="DEBUG" additivity="false">
            <appender-ref ref="Console" />
        </logger>
        <logger name="java.sql.PreparedStatement" level="DEBUG" additivity="false">
            <appender-ref ref="Console" />
        </logger>
        <logger name="java.sql.ResultSet" level="DEBUG" additivity="false">
            <appender-ref ref="Console" />
        </logger>
        <logger name="org.quartz" level="ERROR" additivity="false">
            <appender-ref ref="Console" />
        </logger>
        <!-- 设置打印sql语句配置结束 -->
 
        <!--建立一个默认的root的logger-->
        <root level="${output_log_level}">
            <appender-ref ref="RollingFile"/>
            <appender-ref ref="Console"/>
            <appender-ref ref="InfoFile"/>
            <appender-ref ref="WarnFile"/>
            <appender-ref ref="ErrorFile"/>
        </root>
    </loggers>
 
</configuration>
新log4j2配置log4j2.xml

 

以上就是log4j2的基本配置,然而我项目有一些日志是记录到数据库里的,之前使用的是log4j方式插入数据库,现在升级成log4j2后不知道怎么配置了,网上找的博客资料都很垃圾,我运行不起来。

实际上官网(https://logging.apache.org/log4j/2.x/manual/appenders.html#JDBCAppender)已经给出详细的插入数据库配置方式了,分为三种,一种是DataSource+jndi,一种是ConnectionFactory+class,还有一种是PoolingDriver+DriverManager,网上的资料一般都是用的第二种方式,于是我也打算用第二种,但官网给的方式二引入的是org.apache.commons.dbcp,而我项目用的是com.alibaba.druid,不想引入过多的包,打算还是用druid,折腾了好几天从网上抄袭的例子都无法成功,今天才想起来看druid源码,说来也巧,今天竟然找到了破解方法。。。(ps:感觉百度现在垃圾的不行了,垃圾广告csdn排名被提高不说,平时工作日搜索也全是抄袭博客园的排名网站。。。)

直接贴代码:

package com.dataSourcer;

import com.alibaba.druid.pool.DruidDataSource;
import com.alibaba.druid.pool.DruidDataSourceFactory;

import java.sql.Connection;
import java.sql.SQLException;
import java.util.Properties;

import com.test.adminapi.util.Config;

/**
 * Log4j2 ConnectionFactory
 */
public class ConnectionFactory {

    private DruidDataSource dataSource;

    private ConnectionFactory() {
        Properties properties = new Properties();        
        properties.setProperty("url", "jdbc.url");
        properties.setProperty("username", "jdbc.username");
        properties.setProperty("password", "jdbc.password");

        try {
            dataSource = (DruidDataSource) DruidDataSourceFactory.createDataSource(properties);
        } catch (Exception e) {
            try {
                if (dataSource != null) {
                    /* Druid数据库源对象产生失败后,取消初始化 */
                    dataSource.close();
                }
            } catch (Exception e2) {
            }
        }
    }

    public static Connection getDatabaseConnection() throws SQLException {
        return Singleton.INSTANCE.dataSource.getConnection();
    }

    private static interface Singleton {
        final ConnectionFactory INSTANCE = new ConnectionFactory();
    }
}

而log4j2.xml中关于插入数据库的配置为JDBC,内容如下:

        <!-- =======================================打印DEBUG级别的日志到文件======================================= -->
        <JDBC name="DbMysql" tableName="t_log4j" >
            <ConnectionFactory class="com.dataSourcer.ConnectionFactory" method="getDatabaseConnection"/>
            <Column name="level" pattern="%level"/>
            <Column name="class" pattern="%C"/>
            <Column name="function" pattern="%M"/>
            <Column name="line" pattern="%L"/>
            <Column name="message" pattern="%m" />
        </JDBC>

完整的log4j2.mxl配置为:

<?xml version="1.0" encoding="UTF-8"?>
<!--日志级别以及优先级排序: OFF > FATAL > ERROR > WARN > INFO > DEBUG > TRACE > ALL -->
<!--Configuration后面的status,这个用于设置log4j2自身内部的信息输出,可以不设置,当设置成trace时,你会看到log4j2内部各种详细输出-->
<!--monitorInterval:Log4j能够自动检测修改配置 文件和重新配置本身,设置间隔秒数-->
<configuration status="WARN" monitorInterval="1800">
 
    <Properties>
        <!-- ==============================================公共配置============================================== -->
        <!-- 设置日志文件的目录名称 -->
        <property name="logFileName">pc.admin</property>
 
        <!-- 日志默认存放的位置,可以设置为项目根路径下,也可指定绝对路径 -->
        <!-- 存放路径一:通用路径,window平台 -->
        <!-- <property name="basePath">d:/logs/${logFileName}</property> -->
        <!-- 存放路径二:web工程专用,java项目没有这个变量,需要删掉,否则会报异常,这里把日志放在web项目的根目录下 -->
        <!-- <property name="basePath">${web:rootDir}/${logFileName}</property> -->
        <!-- 存放路径三:web工程专用,java项目没有这个变量,需要删掉,否则会报异常,这里把日志放在tocmat的logs目录下 -->
        <property name="basePath">/data/logs/${logFileName}</property>
 
        <!-- 控制台默认输出格式,"%-5level":日志级别,"%l":输出完整的错误位置,是小写的L,因为有行号显示,所以影响日志输出的性能 -->
        <property name="console_log_pattern">%d{yyyy-MM-dd HH:mm:ss.SSS} [%-5level] %l - %m%n</property>
        <!-- 日志文件默认输出格式,不带行号输出(行号显示会影响日志输出性能);%C:大写,类名;%M:方法名;%m:错误信息;%n:换行 -->
        <!-- <property name="log_pattern">%d{yyyy-MM-dd HH:mm:ss.SSS} [%-5level] %C.%M - %m%n</property> -->
        <!-- 日志文件默认输出格式,另类带行号输出(对日志输出性能未知);%C:大写,类名;%M:方法名;%L:行号;%m:错误信息;%n:换行 -->
        <property name="log_pattern">%d{yyyy-MM-dd HH:mm:ss.SSS} [%-5level] %C.%M[%L line] - %m%n</property>
 
        <!-- 日志默认切割的最小单位 -->
        <property name="every_file_size">20MB</property>
 
 
        <!-- ============================================Error级别日志============================================ -->
        <!-- Error日志默认压缩路径,将超过指定文件大小的日志,自动存入按"年月"建立的文件夹下面并进行压缩,作为存档 -->
        <property name="error_filePattern">${basePath}/%d{yyyy-MM}/error-%d{yyyy-MM-dd}-%i.log.gz</property>
        <!-- Error日志默认同一文件夹下可以存放的数量,不设置此属性则默认为7个 -->
        <property name="error_max">100</property>
        <!-- 日志默认同类型日志,多久生成一个新的日志文件,这个配置需要和filePattern结合使用;
                        如果设置为1,filePattern是%d{yyyy-MM-dd}到天的格式,则间隔一天生成一个文件
                        如果设置为12,filePattern是%d{yyyy-MM-dd-HH}到小时的格式,则间隔12小时生成一个文件 -->
        <property name="error_timeInterval">1</property>
        <!-- 日志默认同类型日志,是否对封存时间进行调制,若为true,则封存时间将以0点为边界进行调整,
                    如:现在是早上3am,interval是4,那么第一次滚动是在4am,接着是8am,12am...而不是7am -->
        <property name="error_timeModulate">true</property>
 
        <!-- ============================================控制台显示控制============================================ -->
        <!-- 控制台显示的日志最低级别 -->
        <property name="console_print_level">ERROR</property>
    </Properties>
 
    <!--定义appender -->
    <appenders>
        <!-- =======================================用来定义输出到控制台的配置======================================= -->
        <Console name="Console" target="SYSTEM_OUT">
            <!-- 设置控制台只输出level及以上级别的信息(onMatch),其他的直接拒绝(onMismatch)-->
            <ThresholdFilter level="${console_print_level}" onMatch="ACCEPT" onMismatch="DENY"/>
            <!-- 设置输出格式,不设置默认为:%m%n -->
            <PatternLayout pattern="${console_log_pattern}"/>
        </Console>
 
        <!-- =======================================打印ERROR级别的日志到文件======================================= -->
        <RollingFile name="ErrorFile" fileName="${basePath}/api.admin.log" filePattern="${error_filePattern}">
            <PatternLayout pattern="${log_pattern}"/>
            <Policies>
                <TimeBasedTriggeringPolicy interval="${error_timeInterval}" modulate="${error_timeModulate}"/>
                <SizeBasedTriggeringPolicy size="${every_file_size}"/>
            </Policies>
            <DefaultRolloverStrategy max="${error_max}" />
            <Filters>
                <ThresholdFilter level="FATAL" onMatch="DENY" onMismatch="NEUTRAL"/>
                <ThresholdFilter level="ERROR" onMatch="ACCEPT" onMismatch="DENY"/>
            </Filters>
        </RollingFile>        
        
        
        <!-- =======================================打印DEBUG级别的日志到文件======================================= -->
        <RollingFile name="ExDebugFile" fileName="${basePath}/api.ex.log" filePattern="${basePath}/%d{yyyy-MM}/api.ex.log.%d{yyyy-MM-dd-HH}">
            <PatternLayout pattern="%d{yyyy-MM-dd HH:mm:ss.SSS} [%p] %C.%M[第%L行] - %m%n"/>
            <Policies>
                <TimeBasedTriggeringPolicy interval="1" modulate="true"/>
                <SizeBasedTriggeringPolicy size="50MB"/>
            </Policies>
            <DefaultRolloverStrategy max="200" />
            <Filters>
                <ThresholdFilter level="DEBUG" onMatch="ACCEPT" onMismatch="DENY"/>
            </Filters>
        </RollingFile>
        <!-- =======================================打印DEBUG级别的日志到文件======================================= -->
        <RollingFile name="WxDebugFile" fileName="${basePath}/api.wx.log" filePattern="${basePath}/%d{yyyy-MM}/api.wx.log.%d{yyyy-MM-dd-HH}">
            <PatternLayout pattern="%d{yyyy-MM-dd HH:mm:ss.SSS} [%-5level] %C.%M[第%L行] - %m%n"/>
            <Policies>
                <TimeBasedTriggeringPolicy interval="1" modulate="true"/>
                <SizeBasedTriggeringPolicy size="50MB"/>
            </Policies>
            <DefaultRolloverStrategy max="200" />
            <Filters>
                <ThresholdFilter level="DEBUG" onMatch="ACCEPT" onMismatch="DENY"/>
            </Filters>
        </RollingFile>
        <!-- =======================================打印DEBUG级别的日志到文件======================================= -->
        <RollingFile name="MailDebugFile" fileName="${basePath}/api.mail.log" filePattern="${basePath}/%d{yyyy-MM}/api.mail.log.%d{yyyy-MM-dd-HH}">
            <PatternLayout pattern="%d{yyyy-MM-dd HH:mm:ss.SSS} [%-5level] %C.%M[第%L行] - %m%n"/>
            <Policies>
                <TimeBasedTriggeringPolicy interval="1" modulate="true"/>
                <SizeBasedTriggeringPolicy size="50MB"/>
            </Policies>
            <DefaultRolloverStrategy max="200" />
            <Filters>
                <ThresholdFilter level="DEBUG" onMatch="ACCEPT" onMismatch="DENY"/>
            </Filters>
        </RollingFile>
        <!-- =======================================打印DEBUG级别的日志到文件======================================= -->
        <RollingFile name="SmsDebugFile" fileName="${basePath}/api.sms.log" filePattern="${basePath}/%d{yyyy-MM}/api.sms.log.%d{yyyy-MM-dd-HH}">
            <PatternLayout pattern="%d{yyyy-MM-dd HH:mm:ss.SSS} [%-5level] %C.%M[第%L行] - %m%n"/>
            <Policies>
                <TimeBasedTriggeringPolicy interval="1" modulate="true"/>
                <SizeBasedTriggeringPolicy size="50MB"/>
            </Policies>
            <DefaultRolloverStrategy max="200" />
            <Filters>
                <ThresholdFilter level="DEBUG" onMatch="ACCEPT" onMismatch="DENY"/>
            </Filters>
        </RollingFile>
        <!-- =======================================打印DEBUG级别的日志到文件======================================= -->
        <RollingFile name="PsDebugFile" fileName="${basePath}/api.ps.log" filePattern="${basePath}/%d{yyyy-MM}/api.ps.log.%d{yyyy-MM-dd-HH}">
            <PatternLayout pattern="%d{yyyy-MM-dd HH:mm:ss.SSS} [%-5level] %C.%M[第%L行] - %m%n"/>
            <Policies>
                <TimeBasedTriggeringPolicy interval="1" modulate="true"/>
                <SizeBasedTriggeringPolicy size="50MB"/>
            </Policies>
            <DefaultRolloverStrategy max="200" />
            <Filters>
                <ThresholdFilter level="DEBUG" onMatch="ACCEPT" onMismatch="DENY"/>
            </Filters>
        </RollingFile>
        <!-- =======================================打印DEBUG级别的日志到文件======================================= -->
        <JDBC name="DbMysql" tableName="t_log4j" >
            <ConnectionFactory class="com.dataSourcer.ConnectionFactory" method="getDatabaseConnection"/>
            <Column name="level" pattern="%level"/>
            <Column name="class" pattern="%C"/>
            <Column name="function" pattern="%M"/>
            <Column name="line" pattern="%L"/>
            <Column name="message" pattern="%m" />
        </JDBC>
    </appenders>
 
    <!--定义logger,只有定义了logger并引入的appender,appender才会生效-->
    <loggers> 
 
        <logger name="ex.debug" level="DEBUG">
            <appender-ref ref="ExDebugFile" />
        </logger> 
        <logger name="ps.debug" level="DEBUG">
            <appender-ref ref="PsDebugFile" />
        </logger>
        <logger name="wx.debug" level="DEBUG">
            <appender-ref ref="WxDebugFile" />
        </logger>
        <logger name="mail.debug" level="DEBUG">
            <appender-ref ref="MailDebugFile" />
        </logger>
        <logger name="sms.debug" level="DEBUG">
            <appender-ref ref="SmsDebugFile" />
        </logger>
        <logger name="db.debug" level="DEBUG">
            <appender-ref ref="DbMysql" />
        </logger>   

 
        <!--建立一个默认的root的logger-->
        <root level="ERROR">
            <appender-ref ref="Console"/>
        </root>
    </loggers>
 
</configuration>
完整log4j2.xml配置

 


 

2022.10.10补充-----

今天另一个项目需要把mapper中执行的sql语句也保存到文件中,原log4j中直接在log4j.properties中配置的mapper相关(见上面配置),但在log4j2.xml中直接配置却始终不起作用。最后发现需要在spring-context.xml中配置sqlSessionFactory的configLocation属性,相应代码如下:

   <!-- spring-context.xml 中 -->
    <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
        <property name="dataSource" ref="dataSource" />
        <property name="configLocation" value="classpath:mybatis-config.xml"></property>
        <property name="mapperLocations">
            <list>
                <value>classpath:com/test/mapper/*.xml</value>
            </list>
        </property>
    </bean>

 

其中configLocation和mapperLocations有的说只保留一个即可,我这里两个都保留貌似正常。

mapperLocation原来已经存在,本次新添加了configLocation配置,同时需要添加新配置文件mybatis-config.xml,内容如下:

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
    <settings>
        <!-- 表示使用log4j2 -->
        <setting name="logImpl" value="LOG4J2" />
    </settings>
</configuration>

 

至此升级完成,准备发布实战~~

 

参考文章:

log4j2官网配置jdbc:https://logging.apache.org/log4j/2.x/manual/appenders.html#JDBCAppender

log4j2配置druid插入数据库:https://blog.csdn.net/qq_23543983/article/details/80687643

 

posted @ 2022-08-27 18:52  一 定 会 去 旅 行  阅读(912)  评论(0编辑  收藏  举报