一.背景
1.项目原来使用jdk6、tomcat6、maven2.2日志使用log4j2没有任何问题,由于升级项目jdk8、tomcat8、maven3.2.5之后日志一行都不出并且没有任何保存信息。
2.项目中是通过代码配置加载log4j2的,servlet2.5以上版本会自己加载log4j2.xml无需配置
public class InitListener implements ServletContextListener {
private Logger log = LogManager.getLogger(InitListener.class);
public void contextInitialized(ServletContextEvent arg0) {
try {
//加载log4j属性
ConfigurationSource source = new ConfigurationSource(this.getClass().getResourceAsStream("/log4j2.xml"));
Configurator.initialize(null, source);
} catch (Throwable e) {
log.error("exception:",e);
}
}
public void contextDestroyed(ServletContextEvent arg0) {
}
}
二、项目现有日志方式
1.commons-logging 是通过LogFactory获取log对象
org.apache.commons.logging.Log log = org.apache.commons.logging.LogFactory.getLog(this.class);
2.log4j 直接使用log4j提供的Logger对象
org.apache.log4j.Logger log = org.apache.log4j.Logger.getLogger(this.class);
3.slf4j(强烈推荐使用此种方式实现)通过LoggerFactory获取log对象
org.slf4j.Logger logger = org.slf4j.LoggerFactory.getLogger(this.class)
发现该有的都有了唯独没有log4j2,但是项目还是用的log4j2.xml在打日志。log4j2通过LogManager获取log对象,log4j2采用异步输出,并且能通过配置实现自动删除日志。
org.apache.logging.log4j.Logger logCostTime = org.apache.logging.log4j.LogManager.getLogger("costTime")
注意4中log获取日志对象方式都是不一样的
三、升级到log4j2
现有代码无需更改,直接桥接到log4j2即可。
1.删除commons-logging、log4j 、slf4j 的maven文件(如果间接依赖进来的exclusion掉)使用桥接的方式
<!-- log4j2必备核心包 log4j-api 、log4j-core-->
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-api</artifactId>
<version>2.13.3</version>
</dependency>
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-core</artifactId>
<version>2.13.3</version>
</dependency>
<!--桥接 Log4j转Log4j2 -->
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-1.2-api</artifactId>
<version>2.13.3</version>
</dependency>
<!--桥接commons-logging转Log4j2-->
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-jcl</artifactId>
<version>2.13.3</version>
</dependency>
<!-- 桥接:告诉Slf4j使用Log4j2 -->
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-slf4j-impl</artifactId>
<version>2.13.3</version>
</dependency>
四、log4j2.xml配置文件
<?xml version="1.0" encoding="UTF-8"?>
<!-- status="OFF",可以去掉,它的含义为是否记录log4j2本身的event信息,默认是OFF -->
<configuration status="OFF">
<!-- 定义下面的引用名 -->
<Properties>
<!--linux目录 /logs -->
<!--<property name="basePath">/logs</property>-->
<property name="basePath">${sys:catalina.home}/logs/webchat</property>
<property name="rolling_pattern">%d{yyyy-MM-dd}-%i.gz</property>
<property name="every_file_size">10MB</property><!-- 日志切割的最小单位 -->
<property name="log_pattern">%d{yyyy-MM-dd HH:mm:ss z} %-5level %class{36} %L %M - %msg%xEx%n</property>
</Properties>
<!--先定义所有的appender -->
<appenders>
<!--这个输出控制台的配置 -->
<Console name="Console" target="SYSTEM_OUT" >
<!--控制台只输出level及以上级别的信息(onMatch),其他的直接拒绝(onMismatch) -->
<ThresholdFilter level="trace" onMatch="ACCEPT" onMismatch="DENY" />
<!--这个都知道是输出日志的格式 -->
<PatternLayout pattern="${log_pattern}" charset="UTF-8"/>
</Console>
<!--这个会打印出所有的信息,每次大小超过size,则这size大小的日志会自动存入按年份-月份建立的文件夹下面并进行压缩,作为存档 -->
<!-- 按月生成归档日志,可以使用 filePattern="logs/$${date:yyyy-MM}/app-%d{MM-dd-yyyy}-%i.log.gz" -->
<RollingFile name="RollingFile" fileName="${basePath}/sinter.log"
filePattern="${basePath}/sinter.log${rolling_pattern}" >
<PatternLayout pattern="${log_pattern}" charset="UTF-8"/>
<SizeBasedTriggeringPolicy size="${every_file_size}" />
</RollingFile>
<!--fileName com.item.interceptor.IndexInterceptor.java 方法日志 -->
<RollingFile name="InterceptorRollingFile" fileName="${basePath}/interceptor.log"
filePattern="${basePath}/item.log${rolling_pattern}" >
<PatternLayout pattern="${log_pattern}" charset="UTF-8"/>
<SizeBasedTriggeringPolicy size="${every_file_size}" />
</RollingFile>
<!--fileName com/item/controller/login/LoginController.java sendCode方法日志 -->
<!-- <File name="sendCodeFile" fileName="${basePath}/sendCode.log" append="false">
<PatternLayout pattern="${log_pattern}"/>
</File>-->
<RollingFile name="sendCodeRollingFile" fileName="${basePath}/sendCodeRollingFile.log" append="true"
filePattern="${basePath}/sendCodeRollingFile.log${rolling_pattern}">
<PatternLayout pattern="${log_pattern}" charset="UTF-8"/>
<Policies>
<SizeBasedTriggeringPolicy size="${every_file_size}"/>
<!-- 每天凌晨0点 匹配的所有文件在10天或更长时间内将在翻滚时间删除。-->
<CronTriggeringPolicy schedule="0 0 0 * * ?"/>
</Policies>
<DefaultRolloverStrategy>
<Delete basePath="${basePath}" maxDepth="2">
<IfFileName glob="*.gz" />
<IfLastModified age="10d" />
</Delete>
</DefaultRolloverStrategy>
</RollingFile>
</appenders>
<!--然后定义logger,只有定义了logger并引入的appender,appender才会生效 -->
<loggers>
<!--建立一个默认的root的logger,需要在root的level中指定输出的级别, -->
<root level="INFO" >
<appender-ref ref="RollingFile" />
<appender-ref ref="Console" />
</root>
<!--将logger中的 additivity 属性配置为 false,则这个logger不会将日志流反馈到root中。-->
<Logger name="sendCodeLogger" additivity="false" level="INFO">
<!--<appender-ref ref="sendCodeFile" level="INFO" />-->
<appender-ref ref="sendCodeRollingFile" level="INFO" />
</Logger>
<Logger name="interceptorLogger" additivity="false" level="INFO">
<appender-ref ref="InterceptorRollingFile" level="INFO" />
</Logger>
</loggers>
</configuration>