写在前面的话:
作为《Java程序员修炼之道》博文的第一个主题Logging,我计划中按照如下三篇来写:
- Logback的简单介绍和配置
- 在Java代码中如何使用SLF4J来写日志以及写日志的要点
- 作为一个程序员,在日常工作中如何分析和挖掘Log。
1. 缘起
写代码中的日志是一个除了用代码实现功能之外最基础最基础的一个技能了,是一个必须掌握的技能。但是目前为止,关于如何写日志的文章和书籍还是不多。
1.1 写日志的必要性
碰到QA提的一个bug的时候,我见识过两种方式的答复:a)请给我重现步骤和重现数据;b)把当时的日志给我。答复前者的,一般需要花很多时间去找问题出现在那里,如果是别人开发的模块的话,花费的时间更多。答复后者的,一般能很快的找到出问题的点,然后就可以开始进入修复的流程。
从概念上来说:日志是一个可运行的、可维护的软件的基础组成部分;通过日志,我们可以了解软件系统在运行中的实时状态,历史状态和异常状态等。一个没有良好日志的软件是所有人的噩梦。
如果你不想给自己找麻烦,你还是把日志好好写写。
1.2 为什么选Logback?
原因有俩:
- 我最近几年用的都是Logback
- 在前几天Log4j 2.0出来之前,logback的Logger用得比较爽
2. 如何配置Logback
2.1 Logback简单介绍
简单来说就是Log4j 1流行了,发现有一些问题是无法解决的,于是又出来了Logback,在Log4j的基础上提升了性能,提高了功能等等。不过前几天又出来了Log4j 2,据说是相对于Logbak来说又提升了性能提高了功能。
2.2 关于SLF4J和Logback
SLF4J(slf4j.org)又称Simple Logging Facade for Java,是一个通用的logging接口,它试图一统Logging框架的天下,兼容了(Log4j 1, java.util.logging和Jakarta Commons Logging)这三个最流行的Logging框架。Logback就是SLF4J的默认实现。
2.3 依赖包导入
2.3.1 一般程序
Maven版
<groupId>ch.qos.logback</groupId>
<artifactId>logback-classic</artifactId>
<version>1.1.2</version>
</dependency>
非Maven版
把http://logback.qos.ch/dist/logback-1.1.2.zip下载下来之后,在其根目录下有logback-core-1.1.2.jar和logback-classic-1.1.2.jar这俩个Jar,在logback-examples\lib下有slf4j-api-1.7.6.jar这个jar,把这三个Jar添加到你的代码包路径中。
2.3.2 非独立运行程序
如果你做的是一个Lib或者API,那么你就不应该依赖于具体的slf4j实现。所以你对logback的依赖应该是在运行测试代码的时候,具体实现方式如下文所示:
Maven版
<groupId>org.slf4j</groupId>
<artifactId>slf4j-api</artifactId>
<version>1.7.6</version>
</dependency>
<dependency>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-classic</artifactId>
<version>1.1.2</version>
<scope>test</scope>
</dependency>
非Maven版
在发布的时候不要把只有Logback才用到jar打包进你的发布程序里面。(如果觉得绕口再多读两遍)
2.4 与遗留Logging框架兼容
目前行业除了Logback之外,广泛使用的还有其他四种Logging框架:
- Log4j 1 (http://logging.apache.org/log4j/1.2/)
- Log4j 2 (http://logging.apache.org/log4j/2.x/)
- java.util.logging (http://docs.oracle.com/javase/8/docs/api/java/util/logging/package-summary.html)
- Apache commons Logging (http://commons.apache.org/proper/commons-logging/)
Log4j 2因为是刚出来的,目前SLF4J对其的兼容性还未知,对于其他的三种框架,SLF4J都提供了兼容性的支持。下面介绍了如何让Logbak兼容这些框架,另外,也可以阅读官方说明:http://www.slf4j.org/legacy.html
2.4.1 兼容Log4j 1和Apache Commons Logging
SLF4J对于Log4J 1和Apache commons Logging的支持方式是提供了实现Log4j和Apache commons Logging接口的SLF4J实现。使用方式是
- 去除对Log4J和Apache commons Logging的Jar包的引用
- 引入SLF4J的对应接口的实现包。
2.4.1.1 移除引用
<groupId>org.springframework.ldap</groupId>
<artifactId>spring-ldap-core</artifactId>
<exclusions>
<exclusion>
<artifactId>commons-logging</artifactId>
<groupId>commons-logging</groupId>
</exclusion>
</exclusions>
</dependency>
如何找到哪些第三方包引用了Log4j或者Apache commons Logging呢?有俩个方法:
- 使用 mvn dependency:tree 命令,如下图所示,可以看出需要在org.springframework.ldap:spring-ldap-core中排除掉对Apache commons Logging的引用。
- 第二种方式是使用Eclipse的m2e Maven插件。如下图所示,打开pom.xml文件后,选择Dependency Hierarchy标签,然后在Filter中输入logging或者log4j进行过滤,在左侧的Dependency Hierarchy中使用右键菜单就可以自动过滤了。
2.4.1.2 Maven导入对应的SLF4J实现包
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>log4j-over-slf4j</artifactId>
<version>1.7.6</version>
</dependency>
<!-- Apache commons Logging 的SLF4J实现 -->
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>jcl-over-slf4j</artifactId>
<version>1.7.6</version>
</dependency>
2.4.1.3 非Maven版导入对应的SLF4J实现包
直接删除掉log4j-1.**.jar和commons-logging-1.**.jar文件,把http://slf4j.org/dist/slf4j-1.7.6.zip下载下来,把压缩包里的log4j-over-slf4j-1.7.6.jar或者(和)jcl-over-slf4j-1.7.6.jar文件放到classpath中。
2.4.2 兼容java.util.logging
SLF4J的jul-to-slf4j模块实现了一个java.util.logging handler,该handler会把对java.util.logging的调用都转化成对SLF4J实现的调用。所以需要以下俩个步骤:
- 导入jul-to-slf4j模块
- 启用jul-to-slf4j模块
2.4.2.1 导入jul-to-slf4j模块maven版
<groupId>org.slf4j</groupId>
<artifactId>jul-to-slf4j</artifactId>
<version>1.7.6</version>
</dependency>
2.4.2.2 导入jul-to-slf4j模块非Maven版
把http://slf4j.org/dist/slf4j-1.7.6.zip下载下来,把压缩包里的jul-to-slf4j-1.7.6.jar放到classpath中。
2.4.2.3 启用jul-to-slf4j模块
在logging.properties中添加如下一行:
2.5 Logback 配置文件简介
2.5.1 Logback配置文件名称、位置和编写策略
Logback 会按照如下的顺序在classpath中读取配置文件,如果读取到任何一个,则停止继续寻找。
- logback.groovy 这个是使用groovy语法的配置文件
- logback-test.xml
- logback.xml
- 输出格式 %d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n
- 输出方向:System.out
- 输出级别:Debug
- 在src/test/resources目录中放置logback-test.xml,该配置文件中log的输出是到输出到console,对于本应用的代码是debug级别,对于其他的是info级别。
- 在src/main/resources目录中放置logback.xml,该配置文件中log的输出是输出到文件中,该文件每日滚动压缩打包备份,对于本应用的代码是info级别,对于其他的是warn级别。
- 对于web项目,logback.xml放置到WEB-INF/classes目录下,配置方式建议参考上面说的
- 对于其他项目,放置到应用启动运行时的classpath根目录下
2.5.2 Logback配置文件示例
<configuration scan="true" scanPeriod="30 seconds">
<!--Appendar详解: http://logback.qos.ch/manual/appenders.html#RollingFileAppender -->
<appender name="FILE" class="ch.qos.logback.core.rolling.RollingFileAppender">
<!-- 当前Log文件名 -->
<file>ldap-pwd.log</file>
<rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
<!-- 非当天的Log文件压缩备份为 archive/ldap-pwd.2014-08-10.zip -->
<fileNamePattern>archive/ldap-pwd.%d{yyyy-MM-dd}.zip</fileNamePattern>
<!-- 超过30天的备份文件会被删除 -->
<maxHistory>30</maxHistory>
</rollingPolicy>
<layout class="ch.qos.logback.classic.PatternLayout">
<!-- 格式说明:http://logback.qos.ch/manual/layouts.html#ClassicPatternLayout -->
<Pattern>%d [%thread] %-5level %40logger{40} - %msg%n</Pattern>
</layout>
</appender>
<logger name="cn.justfly.training.logging" level="info" />
<root level="warn">
<appender-ref ref="FILE" />
</root>
</configuration>
<configuration>
<appender name="Console" class="ch.qos.logback.core.ConsoleAppender">
<encoder>
<pattern>%d [%thread] %-5level %40logger{40} - %msg%n</pattern>
</encoder>
</appender>
<logger name="cn.justfly.training.logging" level="debug" />
<root level="info">
<appender-ref ref="Console" />
</root>
</configuration>