zoukankan      html  css  js  c++  java
  • Log4j各级别日志重复打印的问题

    今天在配置Log4j日志的时候,发现日志重复打印的问题。网上查了很多资料,发现介绍Log4j配置的文章数量不少,但提到这个问题的文章却寥寥,解决了自己的问题以后,赶紧记录一下。

    原文地址:http://www.jianshu.com/p/7f0a1121a6df

    本文基于log4j 1.2.17版本进行说明

    <dependency>
          <groupId>log4j</groupId>
          <artifactId>log4j</artifactId>
          <version>1.2.17</version>
    </dependency>
    

    一、问题描述

    先来看一下日志配置文件:

    #root日志
    log4j.rootLogger=INFO,stdout,info,warn,error
    
    #控制台日志
    log4j.appender.stdout=org.apache.log4j.ConsoleAppender
    log4j.appender.stdout.Target=System.out
    log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
    log4j.appender.stdout.layout.ConversionPattern=%d %-5p %c{1}:%L - %m%n
    
    #info级别日志
    log4j.appender.info=org.apache.log4j.RollingFileAppender
    log4j.appender.info.Threshold=INFO
    log4j.appender.info.File=/home/info.log
    log4j.appender.info.MaxFileSize=200MB
    log4j.appender.info.MaxBackupIndex=5
    log4j.appender.info.layout=org.apache.log4j.PatternLayout
    log4j.appender.info.layout.ConversionPattern=%d %-5p %l - %m%n
    
    #warn级别日志
    log4j.appender.warn=org.apache.log4j.RollingFileAppender
    log4j.appender.warn.Threshold=WARN
    log4j.appender.warn.File=/home/warn.log
    log4j.appender.warn.MaxFileSize=200MB
    log4j.appender.warn.MaxBackupIndex=5
    log4j.appender.warn.layout=org.apache.log4j.PatternLayout
    log4j.appender.warn.layout.ConversionPattern=%d %-5p %l - %m%n
    
    #error级别日志
    log4j.appender.error=org.apache.log4j.RollingFileAppender
    log4j.appender.error.Threshold=ERROR
    log4j.appender.error.File=/home/error.log
    log4j.appender.error.MaxFileSize=200MB
    log4j.appender.error.MaxBackupIndex=5
    log4j.appender.error.layout=org.apache.log4j.PatternLayout
    log4j.appender.error.layout.ConversionPattern=%d %-5p %l - %m%n
    

    上面这个log4j.properties配置文件是个非常常规的文件,网上大多数讲解Log4j配置信息的文章也都是基于这样一个模板展开的。

    使用这套配置,毫无疑问日志是可以正常打印的,但问题就是log4j.appender.info.Threshold指定的级别表示打印等于或者大于这个级别的日志,这样一来当执行下面代码的时候

    Logger LOG = LoggerFactory.getLogger(Test.class);
    LOG.info("info");
    LOG.warn("warn");
    LOG.error("error");
    

    我们看到的现象是:
    /home/info.log文件中会打印info,warn,error三行日志;
    /home/warn.log文件中会打印warn,error两行;
    /home/error.log文件中只会打印error一行日志。

    上面的结果显然不是我们想要的,因为这样的话相当于info日志中含有所有的日志信息,不但造成冗余,而且也会让warn日志跟error日志显得没有存在的必要。更多的情况下我们希望info日志中只有INFO级别的日志,warn日志中只有WARN级别的日志,同样error日志中也只有ERROR级别的日志。

    二、解决办法

    这里提供两种解决方案:

    1. 重写RollingFileAppender类中的isAsSevereAsThreshold方法。

    默认的isAsSevereAsThreshold方法的实现为:

        public boolean isAsSevereAsThreshold(Priority priority) {
            return this.threshold == null || priority.isGreaterOrEqual(this.threshold);
        }
    

    如果没有设置threshold属性则全部打印,否则打印大于等于threshold属性的日志。
    我们继承该类并重写此方法,如下:

    public class MyLog4jAppender extends RollingFileAppender {
        @Override
        public boolean isAsSevereAsThreshold(Priority priority) {
            return priority != null && this.getThreshold() != null
                    && priority.isGreaterOrEqual(this.getThreshold())
                    && this.getThreshold().isGreaterOrEqual(priority);
        }
    }
    

    并在log4j.properties文件中修改log4j.appender.info=com.xxx.MyLog4jAppender即可。

    1. 使用filter进行日志过滤

    这个其实是Log4j自带的方案,也是推荐方案,不知道为什么网上的资料却很少提到这点。
    把log4j.properties配置文件修改成如下:

    #root日志
    log4j.rootLogger=INFO,stdout,info,warn,error
    
    #控制台日志
    log4j.appender.stdout=org.apache.log4j.ConsoleAppender
    log4j.appender.stdout.Target=System.out
    log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
    log4j.appender.stdout.layout.ConversionPattern=%d %-5p %c{1}:%L - %m%n
    
    #info级别日志
    log4j.appender.info=org.apache.log4j.RollingFileAppender
    log4j.appender.info.Threshold=INFO
    log4j.appender.info.File=/home/info.log
    log4j.appender.info.MaxFileSize=200MB
    log4j.appender.info.MaxBackupIndex=5
    log4j.appender.info.layout=org.apache.log4j.PatternLayout
    log4j.appender.info.layout.ConversionPattern=%d %-5p %l - %m%n
    log4j.appender.info.filter.infoFilter = org.apache.log4j.varia.LevelRangeFilter
    log4j.appender.info.filter.infoFilter.LevelMin=INFO
    log4j.appender.info.filter.infoFilter.LevelMax=INFO
    
    #warn级别日志
    log4j.appender.warn=org.apache.log4j.RollingFileAppender
    log4j.appender.warn.Threshold=WARN
    log4j.appender.warn.File=/home/warn.log
    log4j.appender.warn.MaxFileSize=200MB
    log4j.appender.warn.MaxBackupIndex=5
    log4j.appender.warn.layout=org.apache.log4j.PatternLayout
    log4j.appender.warn.layout.ConversionPattern=%d %-5p %l - %m%n
    log4j.appender.warn.filter.infoFilter = org.apache.log4j.varia.LevelRangeFilter
    log4j.appender.warn.filter.infoFilter.LevelMin=WARN
    log4j.appender.warn.filter.infoFilter.LevelMax=WARN
    
    #error级别日志
    log4j.appender.error=org.apache.log4j.RollingFileAppender
    log4j.appender.error.Threshold=ERROR
    log4j.appender.error.File=/home/error.log
    log4j.appender.error.MaxFileSize=200MB
    log4j.appender.error.MaxBackupIndex=5
    log4j.appender.error.layout=org.apache.log4j.PatternLayout
    log4j.appender.error.layout.ConversionPattern=%d %-5p %l - %m%n
    log4j.appender.error.filter.infoFilter = org.apache.log4j.varia.LevelRangeFilter
    log4j.appender.error.filter.infoFilter.LevelMin=ERROR
    log4j.appender.error.filter.infoFilter.LevelMax=ERROR
    

    通过以上配置模板即可解决各级别日志重复打印的问题。


    今天是1024程序员节,兄弟们high起来~

  • 相关阅读:
    Git Bash关键命令
    一个不需要Log4Net的写日志的简单方法
    未知软件
    Linux
    Linux
    Linux
    Linux
    Linux
    Linux
    Linux
  • 原文地址:https://www.cnblogs.com/lfls/p/7725547.html
Copyright © 2011-2022 走看看