前言
下面的总结基本上来自网络与官方文档,这里做一个记录,方便以后查看
SpringBoot 内部采用的是 Commons Logging进行日志记录,但是默认配置也提供了对常用日志的支持,如: Java Util Logging,Log4j,Log4j2和Logback。每种Logger 都可以通过配置使用控制台或者文件输出日志内容。
slf4j
Simple Logging Facade For Java ,它是一个针对于各类Java日志框架的统一 Facade抽象。Java日志框架众多————常用的有
Java Util Logging,Log4j,Log4j2和Logback,Spring框架使用的是 Jakarta Commons Logging API(JCL)。而 slf4j定义了统一的日志抽象接口,而真正的日志实现则是在运行时决定的–它提供了各类日志框架的binding
Logback
Logback是log4j框架的作者开发的新一代日志框架,它效率更高,能够适应诸多的运行环境,同时天然支持 slf4j
默认情况下SpringBoot的日志框架是 logback,在建立SpringBoot项目时自动导入的 spring-boot-starter 依赖包含了 spring-boot-starter-logging,该依赖内容就是 Spring Boot默认的日志框架。
日志格式
默认的日志输出如下:1
2018-09-05 16:49:46.985  INFO 17416 --- [           main] com.pjmike.logger.LoggerApplication      : Started LoggerApplication in 7.555 seconds (JVM running for 8.677)
输出内容元素具体如下:
- 时间日期 —— 精确到毫秒
- 日志级别 —— ERROR,WARN,INFO,DEBUG,TRACE
- 进程ID
- 分割符 —— ---标识实际日志的开始
- 线程名 —— 方括号括起来(可能会截断控制台输出)
- Logger名 —— 通常使用源代码的类名
- 日志内容
注 Logback没有FATAL级别,它会映射到ERROR
日志输出
SpringBoot 默认为我们输出的日志级别为 INFO,WARN,ERROR。如需要输出更多日志的时候,可以通过以下方法开启
- 命令模式配置: java -jar app.jar --debug=true,这种命令会被SpringBoot解析,且优先级最高。
- 资源文件配置: application.properties配置debug=true即可。该配置只对嵌入式容器,Spring,Hibernate生效,我们自己的项目想要输出DEBUG需要额外配置(配置规则:logging.level.<logger-name>=<level>)
日志输出级别配置
| 1 | logging.level.root = WARN | 
日志输出格式配置
- logging.pattern.console: 定义输出到控制台的格式(不支持 JDK Logger)
- logging.pattern.file:定义输出到文件的格式(不支持 JDK Logger)
颜色编码
如果终端支持 ANSI,默认情况下会给日志上个色,提高可读性,可以在配置文件中设置spring.output.ansi.enabled 来改变默认值:
- ALWAYS:启用 ANSI颜色的输出
- DETECT: 尝试检测 ANSI着色功能是否可用
- NEVER: 禁用 ANSI颜色的输出
编码对照表
| Level | Color | 
|---|---|
| WARN | Yellow | 
| FATAL,ERROR | red | 
| INFO,DEBUG,TRACE | Green | 
文件输出
默认情况下,SpringBoot仅将日志输出到控制台,不会写入到日志文件中,如果除了控制台输出之外还想写日志文件,则需要在 application.properties 设置 loggging.file 或 logging.path属性。
- logging.file: 将日志写入到指定的文件中,默认为相对路径。可以设置成绝对路径 - 1 - logging.file = spring-logger.log 
- logging.path: 将名为 - spring.log写入到指定的文件夹中,如(- /var/log)
一般情况下设置其中一个属性就足够了。
日志文件在达到 10MB时进行分割,产生一个新的日志文件(如: spring.1.log,spring.2.log)。新的日志依旧输出到 spring.log中去,默认情况下会记录 ERROR,WARN,INFO级别消息
- logging.file.max-size: 限制日志文件的大小
- logging.file.max-history: 限制日志保留天数
自定义日志配置
在官网中有如下一段描述:
Since logging is initialized before the ApplicationContext is created, it is not possible to control logging from @PropertySources in Spring @Configuration files. The only way to change the logging system or disable it entirely is via System properties.
由于创建 ApplicationContext 之前就已经初始化日志记录了,因此无法在Spring @Configuration文件中控制来自@PropertySources的日志记录。它并不是必须通过 Spring 的配置文件控制,因此通过系统属性 logging.config和传统的Spring Boot外部配置文件依然可以很好的支持日志控制和管理。
根据不同的日志系统,你可以按如下规则组织配置文件名,就能被正确加载:
- Logback: logback-spring.xml,logback-spring.groovy,logback.xml,logback.groovy
- Log4j: log4j-spring.properties,log4j-spring.xml,log4j.properties,log4j.xml
- Log4j2: log4j2-spring.xml,log4j2.xml
- JDK(Java Util Logging): logging.properties
SpringBoot官方推荐优先使用带有 -spring的文件名为你的日志配置 (比如使用 logback-spring.xml,而不是 logback.xml,logback.xml配置加载过早,比application.properties还早,不适合进行扩展)
Logback配置
SpringBoot 默认的日志框架是 logback,下面我们来一个简单的 logback-spring.xml的例子:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25<?xml version="1.0" encoding="UTF-8"?>
<configuration  scan="true" scanPeriod="60 seconds" debug="false">
    <contextName>logback</contextName>
    <!--输出到控制台-->
    <appender name="console" class="ch.qos.logback.core.ConsoleAppender">
        <encoder>
            <pattern>%d{HH:mm:ss.SSS} %contextName [%thread] %-5level %logger{36} - %msg%n</pattern>
        </encoder>
    </appender>
    <!--输出到文件-->
    <appender name="file" class="ch.qos.logback.core.rolling.RollingFileAppender">
        <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
            <fileNamePattern>logback.%d{yyyy-MM-dd}.log</fileNamePattern>
        </rollingPolicy>
        <encoder>
            <pattern>%d{HH:mm:ss.SSS} %contextName [%thread] %-5level %logger{36} - %msg%n</pattern>
        </encoder>
    </appender>
    <root level="info">
        <appender-ref ref="console" />
        <appender-ref ref="file" />
    </root>
</configuration>
根节点 <configuration>包含的属性:
- scan: 当此属性设置为 true时,配置文件如果发生改变,将会被重新加载,默认值为 true
- scanPeriod: 设置监测配置文件是否有修改的时间间隔,如果没
- debug: 当属性设置为true时,将打印出 logback内部日志信息,实时查看 logback运行状态,默认值为 false.
根节点<configuration>:下面的两个属性,三个子节点
- 属性一:设置上下文名称 - <contextName>- 每个logger都关联到logger上下文,默认上下文名称为“default”。但可以使用设置成其他名字,用于区分不同应用程序的记录。一旦设置,不能修改,可以通过%contextName来打印日志上下文名称 - 1 - <contextName>logback</contextName> 
- 属性二: 设置变量 - <property>- 用来定义变量值的标签, 有两个属性,name和value;其中name的值是变量的名称,value的值时变量定义的值。通过定义的值会被插入到logger上下文中。定义变量后,可以使“${}”来使用变量 - 1 - <property name="log.path" value="/Documents/log" /> 
- 子节点 - <appender>- appender用来格式化日志输出节点,有两个属性,name和class,class用来指定哪种输出策略,常用就是控制台输出策略和文件输出策略 
控制台输出 ConsoleAppender:1
2
3
4
5
6
7
8
9<!--输出到控制台-->
<appender name="console" class="ch.qos.logback.core.ConsoleAppender">
    <filter class="ch.qos.logback.classic.filter.ThresholdFilter">
        <level>ERROR</level>
    </filter>
    <encoder>
        <pattern>%d{HH:mm:ss.SSS} %contextName [%thread] %-5level %logger{36} - %msg%n</pattern>
    </encoder>
</appender>
<ercoder>表示对日志进行编码:
- %d{HH: mm:ss.SSS}——日志输出时间
- %thread——输出日志的进程名字,这在Web应用以及异步任务处理中很有用
- %-5level——日志级别,并且使用5个字符靠左对齐
- %logger{36}——日志输出者的名字
- %msg——日志消息
- %n——平台的换行符
ThresholdFilter为系统定义的拦截器,例如我们用 ThresholdFilter来过滤 ERROR 级别以下的日志不输出到文件中。如果不用记得注释掉,不然你控制台发现没日志
输出到文件 RollingFileAppender:
另一种常见的日志输出到文件,随着应用的运行时间越来越长,日志也会增长的越来越多,将他们输出到同一个文件并非一个好办法.
RolliingFileAppender用于切分文件日志:1
2
3
4
5
6
7
8
9<!--输出到文件-->
<appender name="file" class="ch.qos.logback.core.rolling.RollingFileAppender">
    <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
        <fileNamePattern>logback.%d{yyyy-MM-dd}.log</fileNamePattern>
    </rollingPolicy>
    <encoder>
        <pattern>%d{HH:mm:ss.SSS} %contextName [%thread] %-5level %logger{36} - %msg%n</pattern>
    </encoder>
</appender>
其中重要的是rollingPolicy的定义,上例中
- 子节点二<root>root节点是必选节点,用来指定最基础的日志输出级别,只有一个level属性。 
level:用来设置打印级别,大小写无关:TRACE, DEBUG, INFO, WARN, ERROR, ALL 和 OFF,不能设置为INHERITED或者同义词NULL。
默认是DEBUG。
可以包含零个或多个元素,标识这个appender将会添加到这个logger。
| 1 | <root level="debug"> | 
- 子节点三 <logger>
用来设置某一个包或者具体的某一个类的日志打印级别,以及指定 。 仅有一个name属性,一个可选的level和一个可选的 addtivity属性 
- name: 用来指定受此 logger 约束的某一个包或者具体的某一个类
- level: 用来设置打印级别,大小写无关: TRACE,DEBUG,INFO,WARN,ERROR,ALL和OFF。,还有一个特俗值INHERITED或者同义词NULL,代表强制执行上级的级别。如果未设置此属性,那么当前logger将会继承上级的级别。(这里的上级就是上面提到的 root)
- additivity: 是否向上级logger传递打印信息,默认为true。
logger在实际使用的时候有两种情况:
- 在代码中使用: - 1 
 2
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 public class HelloController {
 private Logger logger = LoggerFactory.getLogger(HelloController.class);
 ("/index")
 public String index() {
 logger.trace("日志输出 trace");
 logger.debug("日志输出 debug");
 logger.info("日志输出 info");
 logger.warn("日志输出 warn");
 logger.error("日志输出 error");
 return "index";
 }
 }
- 在xml文件中使用,例子如下: - 1 
 2
 3- <logger name="com.pjmike.controller.HelloController" level="WARN" additivity="false"> 
 <appender-ref ref="console"/>
 </logger>
多环境日志输出
有时候我们需要在不同的环境(prod:生产环境,test: 测试环境,dev: 开发环境)来定义不同的日志输出,这类情况一般有两种做法:
- 第一种就是为每一种环境都创建一个对应日志配置文件,然后再其对应环境的application.properties指定 logging.cofig属性
- 第二种就是使用扩展属性 springProfile与springProperty
springProfile
<springProfile>标签使我们让配置文件更加灵活,它可以选择性的包含或排除部分配置
2
3
4
5
6
7
8
9
10
11
12
<springProfile name="dev">
<!-- 开发环境时激活 -->
</springProfile>
<springProfile name="dev,test">
<!-- 开发,测试的时候激活-->
</springProfile>
<springProfile name="!prod">
<!-- 当 "生产" 环境时,该配置不激活-->
</springProfile>
例子1
2
3
4
5
6
7
8
9
10
11
12<springProfile name="dev">
        <!--开发环境时激活-->
        <root level="debug">
            <appender-ref ref="console"/>
        </root>
</springProfile>
<springProfile name="test">
        <root level="info">
            <appender-ref ref="file"/>
            <appender-ref ref="console"/>
        </root>
</springProfile>
springProperty
<springProperty>标签可以让我们在 Logback中使用 Spring Environment中的属性。如果想在logback-spring.xml中回读application.properties配置的值时,这是一个非常好的解决方案
2
3
4
5
6
7
8
9
10
11
scope:作用域
name:在 logback-spring.xml 使用的键
source:application.properties 文件中的键
defaultValue:默认值
-->
<springProperty scope="context" name="logName" source="spring.application.name" defaultValue="myapp.log"/>
<appender name="file" class="ch.qos.logback.core.rolling.RollingFileAppender">
<file>logs/${logName}.log</file>
</appender>
小结
以上总结也参考了一些大佬的教程,关于日志配置更多的细节可以参阅 官方文档: https://docs.spring.io/spring-boot/docs/2.0.1.RELEASE/reference/htmlsingle/#boot-features-custom-log-configuration
PS: 有时候我们会觉得在每个类中写Logger会很繁琐,这里推荐一款插件叫做 Lombok ,它里面有一个注解 @Slf4j 能够实现Logger的功能,非常好用,关于Lombok的详细介绍参照文章: http://t.cn/RS0UdrX