• Java Logger 使用总结


    Java Logger 使用总结

    • logger 背景知识
      • logback取代 log4j的理由: http://www.cnblogs.com/lixuwu/p/5804793.html
      • logback、log4j、log4j2三种日志框架性能检测——为什么用log4j2 : https://blog.csdn.net/qq_32250495/article/details/82382052
      • Log4j2的性能为什么这么好?:https://mp.weixin.qq.com/s/ZOkKxYAmRy950lsUcf9RuA
    • Spring Boot 日志
      • Spring Boot 默认使用Logback作为日志记录工具。日志默认输出到控制台但也能输出到文件中。我们通过spring-boot-starter-logging 加入Logback依赖,其实只要我们加入任意的Spring Boot starter 都会默认引入spring-boot-starter-logging,因此 我们不需要分开加入他们
      •  我们可以在application.properties或者application.yml中设置日志的级别、输出的文件名、输出路径、定义console, file 中logging的样式等。
    logging.level.root= WARN
    logging.level.org.springframework.security= DEBUG
    logging.level.org.springframework.web= ERROR
    logging.level.org.hibernate= DEBUG
    
    logging.file = mylogfile.log 
    
    logging.path = concretepage/logs  
    
    logging.pattern.console= %d{yyyy-MMM-dd HH:mm:ss.SSS} %-5level [%thread] %logger{15} - %msg%n
    logging.pattern.file= %d{yyyy-MMM-dd HH:mm:ss.SSS} %-5level [%thread] %logger{15} - %msg%n
      • 如果想使用XML配置Logback,则需要在 resources 路径下创建 logback-spring.xml文件,如果Spring Boot使用了Log4j2 ,则需要创建 log4j2-spring.xml 。
      • Spring Boot 想使用Log4j2 ,则需要修改 Maven 依赖
    <dependency>
       <groupId>org.springframework.boot</groupId>
       <artifactId>spring-boot-starter-web</artifactId>
       <!--使用下面这个和上面排除spring-boot-starter-logging是一样的效果-->
       <exclusions>
          <exclusion>
             <groupId>org.springframework.boot</groupId>
             <artifactId>spring-boot-starter-logging</artifactId>
          </exclusion>
       </exclusions>
    </dependency>
    <!-- log4j2日志 -->
    <dependency>
       <groupId>org.springframework.boot</groupId>
       <artifactId>spring-boot-starter-log4j2</artifactId>
    </dependency>

    使用 logstash-logback-encoder jar 包, 将 logback 日志传给 logstash

    <dependency>
      <groupId>net.logstash.logback</groupId>
      <artifactId>logstash-logback-encoder</artifactId>
      <version>4.11</version>
    </dependency>
        <!-- logstash -->
        <appender name="stash" class="net.logstash.logback.appender.LogstashTcpSocketAppender">
            <destination>logstash.rongziidc.com:5046</destination>
            <!-- encoder is required -->
            <encoder class="net.logstash.logback.encoder.LogstashEncoder" >
                <customFields>{"project":"com.rongziidc.creditorv2","type":"javaLog"}</customFields>
                <timeZone>UTC+8</timeZone>
                <timestampPattern>yyyy-MM-dd HH:mm:ss.SSS</timestampPattern>
            </encoder>
            <connectionStrategy>
                <roundRobin>
                    <connectionTTL>5 minutes</connectionTTL>
                </roundRobin>
            </connectionStrategy>
        </appender>

    logback中动态获取application中配置项 :

    <springProperty scope="context" name="LOG_HOME" source="logback.file"/>

    log4j 的使用

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

    配置log4j , log4j.xml 配置详解

    <?xml version="1.0" encoding="UTF-8"?>
    <!DOCTYPE log4j:configuration PUBLIC "-//log4j/log4j Configuration//EN" "log4j.dtd">
    <log4j:configuration xmlns:log4j="http://jakarta.apache.org/log4j/">
        <!-- 日志输出到控制台 -->
        <appender name="console" class="org.apache.log4j.ConsoleAppender">
            <!-- 日志输出格式 -->
            <layout class="org.apache.log4j.PatternLayout">
                <param name="ConversionPattern" value="[%p][%d{yyyy-MM-dd HH:mm:ss SSS}][%c]-[%m]%n"/>
            </layout>
            <!--过滤器设置输出的级别-->
            <filter class="org.apache.log4j.varia.LevelRangeFilter">
                <!-- 设置日志输出的最小级别 -->
                <param name="levelMin" value="INFO"/>
                <!-- 设置日志输出的最大级别 -->
                <param name="levelMax" value="ERROR"/>
            </filter>
        </appender>
    
        <!-- 输出日志到文件 -->
        <appender name="fileAppender" class="org.apache.log4j.FileAppender">
            <!-- 输出文件全路径名-->
            <param name="File" value="/data/applogs/own/fileAppender.log"/>
            <!--是否在已存在的文件追加写:默认时true,若为false则每次启动都会删除并重新新建文件-->
            <param name="Append" value="false"/>
            <param name="Threshold" value="INFO"/>
            <!--是否启用缓存,默认false-->
            <param name="BufferedIO" value="false"/>
            <!--缓存大小,依赖上一个参数(bufferedIO), 默认缓存大小8K  -->
            <param name="BufferSize" value="512"/>
            <!-- 日志输出格式 -->
            <layout class="org.apache.log4j.PatternLayout">
                <param name="ConversionPattern" value="[%p][%d{yyyy-MM-dd HH:mm:ss SSS}][%c]-[%m]%n"/>
            </layout>
        </appender>
    
        <!-- 输出日志到文件,当文件大小达到一定阈值时,自动备份, FileAppender子类 -->
        <appender name="rollingAppender" class="org.apache.log4j.RollingFileAppender">
            <!-- 日志文件全路径名 -->
            <param name="File" value="/data/applogs/RollingFileAppender.log" />
            <!--是否在已存在的文件追加写:默认时true,若为false则每次启动都会删除并重新新建文件-->
            <param name="Append" value="true" />
            <!-- 保存备份日志的最大个数,默认值是:1  -->
            <param name="MaxBackupIndex" value="10" />
            <!-- 设置当日志文件达到此阈值的时候自动回滚,单位可以是KB,MB,GB,默认单位是KB,默认值是:10MB -->
            <param name="MaxFileSize" value="10KB" />
            <!-- 设置日志输出的样式 -->`
            <layout class="org.apache.log4j.PatternLayout">
                <!-- 日志输出格式 -->
                <param name="ConversionPattern" value="[%d{yyyy-MM-dd HH:mm:ss:SSS}] [%-5p] [method:%l]%n%m%n%n" />
            </layout>
        </appender>
    
        <!-- 日志输出到文件,可以配置多久产生一个新的日志信息文件 -->
        <appender name="dailyRollingAppender" class="org.apache.log4j.DailyRollingFileAppender">
            <!-- 文件文件全路径名 -->
            <param name="File" value="/data/applogs/own/dailyRollingAppender.log"/>
            <param name="Append" value="true" />
            <!-- 设置日志备份频率,默认:为每天一个日志文件 -->
            <param name="DatePattern" value="'.'yyyy-MM-dd'.log'" />
            <!--每分钟一个备份-->
            <!--<param name="DatePattern" value="'.'yyyy-MM-dd-HH-mm'.log'" />-->
            <layout class="org.apache.log4j.PatternLayout">
                <param name="ConversionPattern" value="[%p][%d{HH:mm:ss SSS}][%c]-[%m]%n"/>
            </layout>
        </appender>
    
        <!--
            1. 指定logger的设置,additivity是否遵循缺省的继承机制
            2. 当additivity="false"时,root中的配置就失灵了,不遵循缺省的继承机制
            3. 代码中使用Logger.getLogger("logTest")获得此输出器,且不会使用根输出器
        -->
        <logger name="logTest" additivity="false">
            <level value ="INFO"/>
            <appender-ref ref="dailyRollingAppender"/>
        </logger>
    
        <!-- 根logger的设置,若代码中未找到指定的logger,则会根据继承机制,使用根logger-->
        <root>
            <appender-ref ref="console"/>
            <appender-ref ref="fileAppender"/>
            <appender-ref ref="rollingAppender"/>
            <appender-ref ref="dailyRollingAppender"/>
        </root>
    </log4j:configuration>

    使用 log4j

    import org.apache.log4j.Logger;
    import org.junit.Test;
    
    public class Test {
        private static Logger log = Logger.getLogger("logTest");
        @Test
        public void testLog(){
            log.debug("debug");
            log.error("error");
        }
    }

    log4j2 的使用

    • 添加 Maven 依赖

    <dependency>
      <groupId>org.apache.logging.log4j</groupId>
      <artifactId>log4j-core</artifactId>
      <version>2.13.0</version>
    </dependency>
    <dependency>
      <groupId>org.apache.logging.log4j</groupId>
      <artifactId>log4j-api</artifactId>
      <version>2.13.0</version>
    </dependency>

    配置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="30">
        <!--Appenders节点,常见的有三种子节点:Console、RollingFile、File -->
        <appenders>
            <!--这个输出控制台的配置-->
            <console name="Console" target="SYSTEM_OUT">
                     <!--输出日志的格式-->
                <PatternLayout pattern="[%d{HH:mm:ss:SSS}] [%p] - %l - %m%n"/>
            </console>
    
            <!--文件会打印出所有信息,这个log每次运行程序会自动清空,由append属性决定,这个也挺有用的,适合临时测试用-->
            <File name="log" fileName="log/test.log" append="false">
                <PatternLayout pattern="%d{HH:mm:ss.SSS} %-5level %class{36} %L %M - %msg%xEx%n"/>
            </File>
    
            <!-- 这个会打印出所有的info及以下级别的信息,每次大小超过size,则这size大小的日志会自动存入按年份-月份建立的文件夹下面并进行压缩,作为存档-->
            <RollingFile name="RollingFileInfo" fileName="${sys:user.home}/logs/info.log" filePattern="${sys:user.home}/logs/$${date:yyyy-MM}/info-%d{yyyy-MM-dd}-%i.log">
                <!--控制台只输出level及以上级别的信息(onMatch),其他的直接拒绝(onMismatch)-->       
                <ThresholdFilter level="info" onMatch="ACCEPT" onMismatch="DENY"/>
                <PatternLayout pattern="[%d{HH:mm:ss:SSS}] [%p] - %l - %m%n"/>
                <Policies>
                     <TimeBasedTriggeringPolicy/>
                     <SizeBasedTriggeringPolicy size="100 MB"/>
                </Policies>
            </RollingFile>
    
            <RollingFile name="RollingFileWarn" fileName="${sys:user.home}/logs/warn.log" filePattern="${sys:user.home}/logs/$${date:yyyy-MM}/warn-%d{yyyy-MM-dd}-%i.log">
                <ThresholdFilter level="warn" onMatch="ACCEPT" onMismatch="DENY"/>
                <PatternLayout pattern="[%d{HH:mm:ss:SSS}] [%p] - %l - %m%n"/>
                <Policies>
                     <TimeBasedTriggeringPolicy/>
                     <SizeBasedTriggeringPolicy size="100 MB"/>
                </Policies>
                <!-- DefaultRolloverStrategy属性如不设置,则默认为最多同一文件夹下7个文件 -->
                <DefaultRolloverStrategy max="20"/>
            </RollingFile>
    
            <RollingFile name="RollingFileError" fileName="${sys:user.home}/logs/error.log" filePattern="${sys:user.home}/logs/$${date:yyyy-MM}/error-%d{yyyy-MM-dd}-%i.log">
                <ThresholdFilter level="error" onMatch="ACCEPT" onMismatch="DENY"/>
                <PatternLayout pattern="[%d{HH:mm:ss:SSS}] [%p] - %l - %m%n"/>
                <Policies>
                     <TimeBasedTriggeringPolicy/>
                     <SizeBasedTriggeringPolicy size="100 MB"/>
                </Policies>
            </RollingFile>
         </appenders>
    
         <!--Loggers节点,常见的有两种:Root和Logger.-->
         <!--然后定义logger,只有定义了logger并引入的appender,appender才会生效-->
         <loggers>
             <!--Logger节点用来单独指定日志的形式,比如要为指定包下的class指定不同的日志级别等-->
             <logger name="org.springframework" level="INFO"></logger>
             <logger name="org.mybatis" level="INFO"></logger>
             <!--Root节点用来指定项目的根日志,如果没有指定Logger,那么就会默认使用该Root日志输出-->
             <root level="all">
                 <appender-ref ref="Console"/>
                 <appender-ref ref="RollingFileInfo"/>
                 <appender-ref ref="RollingFileWarn"/>
                 <appender-ref ref="RollingFileError"/>
             </root>
         </loggers>
    </configuration>

    使用 log4j2

    import org.apache.logging.log4j.LogManager;
    import org.apache.logging.log4j.Logger;
    
    private static Logger logger_ = LogManager.getLogger(DateUtils2Joda.class);

    logback

    • 添加 Maven依赖,logback-classic本身实现了SLF4J API

        <dependency>
            <groupId>ch.qos.logback</groupId>
            <artifactId>logback-classic</artifactId>
            <version>1.2.3</version>
        </dependency>

    配置 logback , logback.xml详解

    <?xml version="1.0" encoding="UTF-8"?>
    <configuration>  
      <!-- 彩色日志 , 彩色日志依赖的渲染类 -->
      <conversionRule conversionWord="clr" converterClass="org.springframework.boot.logging.logback.ColorConverter" />
      <conversionRule conversionWord="wex" converterClass="org.springframework.boot.logging.logback.WhitespaceThrowableProxyConverter" />
      <conversionRule conversionWord="wEx" converterClass="org.springframework.boot.logging.logback.ExtendedWhitespaceThrowableProxyConverter" />
      <!-- 彩色日志格式 -->
      <property name="CONSOLE_LOG_PATTERN" value="${CONSOLE_LOG_PATTERN:-%clr(%d{yyyy-MM-dd HH:mm:ss.SSS}){faint} %clr(${LOG_LEVEL_PATTERN:-%5p}) %clr(${PID:- }){magenta} %clr(---){faint} %clr([%15.15t]){faint} %replace(%caller{1}){'\t|Caller.{1}0|\r\n', ''} %clr(:){faint} %m%n${LOG_EXCEPTION_CONVERSION_WORD:-%wEx}}" />
      
      <property name="PROJECT_NAME" value="missyou_app_api"/>
        <property name="LOG_HOME" value="/mnt/logs/" />
      
      <!-- 控制台打印日志的相关配置 --> 
      <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender"> 
        <!-- 日志格式 -->
        <encoder>
                <pattern>${CONSOLE_LOG_PATTERN}</pattern>
            <!--解决乱码问题 -->
            <charset>utf8</charset>
        </encoder>
      </appender>
    
        <appender name="infoLogFile" class="ch.qos.logback.core.rolling.RollingFileAppender">
            <!-- 过滤掉ERROR日志 -->
            <filter class="ch.qos.logback.classic.filter.LevelFilter">
                <level>ERROR</level>
                <onMatch>DENY</onMatch>
                <onMismatch>ACCEPT</onMismatch>
            </filter>
            <encoder>
                <charset>UTF-8</charset>
                <pattern>[%d{yyyy-MM-dd HH:mm:ss.SSS}] [%thread] %-5level %logger{36}: %msg%n</pattern>
            </encoder>
            <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
                <fileNamePattern>${LOG_HOME}/info.%d{yyyy-MM-dd}.log</fileNamePattern>
                <maxHistory>30</maxHistory><!--保存最近30天的日志 -->
            </rollingPolicy>
        </appender>
     
        <appender name="errorLogFile" class="ch.qos.logback.core.rolling.RollingFileAppender">
            <!-- ThresholdFilter:临界值过滤器,过滤掉 TRACE 和 DEBUG 级别的日志 -->
            <filter class="ch.qos.logback.classic.filter.ThresholdFilter">
                <level>error</level>
            </filter>
            <encoder>
                <charset>UTF-8</charset>
                <pattern>[%d{yyyy-MM-dd HH:mm:ss.SSS}] [%thread] %-5level %logger{36}: %msg%n</pattern>
            </encoder>
            <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
                <fileNamePattern>${LOG_HOME}/error.%d{yyyy-MM-dd}.log
                </fileNamePattern>
                <maxHistory>30</maxHistory><!--保存最近30天的日志 -->
            </rollingPolicy>
        </appender>
    
      <logger name="org.hibernate.SQL" level="DEBUG" />
      <!--SpringBoot去掉CONDITIONS EVALUATION REPORT https://blog.csdn.net/YooFale/article/details/84871618-->
      <logger name="org.springframework.boot.autoconfigure" level="error"/>
      
      <!-- 基于dubug处理日志:具体控制台或者文件对日志级别的处理还要看所在appender配置的filter,如果没有配置filter,则使用root配置 -->
      <root level="info">
        <appender-ref ref="STDOUT" />
        <appender-ref ref="infoLogFile" />
        <appender-ref ref="errorLogFile" />
      </root>
    </configuration>

    使用 logback

    private static Logger logging = LoggerFactory.getLogger(TestController.class);
     
    @RequestMapping("/testLogger")
    public void testLogger(int id, String name) {
        logging.info("id:{}, name:{}",id,name);
    }

    slf4j (Simple Logging Facade for Java)

    slf4j 背景知识 官方文档 , github

    slf4j 主要是为了给Java日志提供一个标准、规范的API,意义在于提供接口,具体实现交由其他日志框架
    slf4j 使得应用程序可以在运行时选择绑定到一个特定的日志框架(如:java.util.logging、Log4j、 Logback),这通过在应用程序的类路径中添加对应的日志框架来实现。如果在类路径中配置的日志框架不可用,抽象层就会立刻取消调用日志的相应逻辑
    阿里巴巴Java开发手册中提出:应用中不可直接使用日志系统(Log4j、Logback)中的 API,而应依赖SLF4J 中的 API,使用门面模式的日志框架,有利于维护和各个类的日志处理方式统一
    添加 Maven 依赖

    <dependency>
        <groupId>org.slf4j</groupId>
        <artifactId>slf4j-api</artifactId>
        <version>1.7.2</version>
    </dependency>
    <dependency>
        <groupId>org.slf4j</groupId>
        <artifactId>slf4j-log4j12</artifactId>
        <version>1.7.2</version>
    </dependency>

    配置log4j ,xml配置,可以参见log4j 部分

    ### 设置###
    log4j.rootLogger = debug,stdout,D,E
    
    ### 输出信息到控制抬 ###
    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 = [%-5p] %d{yyyy-MM-dd HH:mm:ss,SSS} method:%l%n%m%n
    
    ### 输出DEBUG 级别以上的日志到=E://logs/error.log ###
    log4j.appender.D = org.apache.log4j.DailyRollingFileAppender
    log4j.appender.D.File = D://logs/log.log
    log4j.appender.D.Append = true
    log4j.appender.D.Threshold = DEBUG
    log4j.appender.D.layout = org.apache.log4j.PatternLayout
    log4j.appender.D.layout.ConversionPattern = %-d{yyyy-MM-dd HH:mm:ss}  [ %t:%r ] - [ %p ]  %m%n
    
    ### 输出ERROR 级别以上的日志到=E://logs/error.log ###
    log4j.appender.E = org.apache.log4j.DailyRollingFileAppender
    log4j.appender.E.File =D://logs/error.log
    log4j.appender.E.Append = true
    log4j.appender.E.Threshold = ERROR
    log4j.appender.E.layout = org.apache.log4j.PatternLayout
    log4j.appender.E.layout.ConversionPattern = %-d{yyyy-MM-dd HH:mm:ss}  [ %t:%r ] - [ %p ]  %m%n

    ava测试类 : 直接使用的是 org.slf4j.Logger ,而不是log4j

    package test;
    import org.slf4j.Logger;
    import org.slf4j.LoggerFactory;
    
    public class Log4jTestDemo {
        private static Logger logger = LoggerFactory.getLogger(Log4jTestDemo.class);
    
        public static void main(String[] args) {
            // 记录debug级别的信息
            logger.debug("This is debug message.");
            // 记录info级别的信息
            logger.info("This is info message.");
            // 记录error级别的信息
            logger.error("This is error message.");
        }
    }
  • 相关阅读:
    ACM TJU 1556
    HDU 1890 Robotie Sort
    Android学习笔记
    HDU 2795
    HDU 1542
    HDU 1698
    POJ 2185
    学习笔记
    HDU 3336
    HDU 3746
  • 原文地址:https://www.cnblogs.com/jijm123/p/15624822.html
Copyright © 2020-2023  润新知