当前主流日志选择slf4j+logback
日志门面采取的就是facade设计模式设计的一组接口应用,SLF4J,即简单日志门面(Simple Logging Facade for Java),不是具体的日志解决方案,它只服务于各种各样的日志系统。在使用SLF4J的时候,不需要在代码中或配置文件中指定你打算使用那个具体的日志系统。
日志实现当然是接口的实现了,Logback是由log4j创始人设计的一个开源日志组件。logback当前分成三个模块:logback-core,logback-classic和logback-access。logback-core是其它两个模块的基础模块。logback-classic是log4j的一个改良版本。
我们先从日志实现分析下手,
JUL(java.util.logger)直接放弃使用,设计简单,不足以支撑开发应用系统。
Log4j是非常流行的日志实现,但是目前作者进行了日志升级就是LogBack,作者在官网声明了logback的很多优点比log4j更强大
Log4j2性能强悍,设计过于复杂,很多开源框架不支持,很容易踩坑。不推荐使用,而且logback的性能足以支撑使用。所以我们采用的日志实现是logback。
日志门面当然是选择logback天然支持的SLF4j。更匹配,性能更加。
springboot采用的同样也是SLF4j+logback。日志框架。
maven项目加入依赖就可以使用了
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-api</artifactId>
<version>1.7.25</version>
</dependency>
<dependency>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-classic</artifactId>
<version>1.2.3</version>
</dependency>
要想在日志中输出主机的IP还需要写一个配置类
package com.zyu.webmoudle.Config;
import ch.qos.logback.classic.pattern.ClassicConverter;
import ch.qos.logback.classic.spi.ILoggingEvent;
import java.net.InetAddress;
import java.net.UnknownHostException;
//获取本机IP
public class IPLogConfig extends ClassicConverter {
@Override
public String convert(ILoggingEvent event) {
try {
return InetAddress.getLocalHost().getHostAddress();
} catch (UnknownHostException e) {
e.printStackTrace();
}
return null;
}
}
下面是一个测试logback.xml的配置文件,注解我应该写得很详细了
<?xml version="1.0" encoding="UTF-8"?>
<!-- scan:当此属性设置为true时,配置文件如果发生改变,将会被重新加载,默认值为true。
scanPeriod:设置监测配置文件是否有修改的时间间隔,如果没有给出时间单位,默认单位是毫秒。当scan为true时,此属性生效。默认的时间间隔为1分钟。
debug:当此属性设置为true时,将打印出logback内部日志信息,实时查看logback运行状态。默认值为false。 -->
<configuration scan="true" scanPeriod="60 seconds" debug="false">
<!-- 上下文变量设置,用来定义变量值,其中name的值是变量的名称,value的值是变量定义的值。
通过<property>定义的值会被插入到logger上下文中。定义变量后,可以使“${}”来使用变量。 -->
<property name="CONTEXT_NAME" value="logback-test"/>
<!--定义日志文件的存储地址 勿在 LogBack 的配置中使用相对路径,一个点表示项目的路径,两个点和项目同级的路径,可以下相对路径,也可以写绝对路径,在-->
<property name="LOG_HOME" value="E:/logs"/>
<!--配置规则类的位置-->
<conversionRule conversionWord="ip" converterClass="com.zyu.webmoudle.Config.IPLogConfig" />
<!-- 上下文名称:<contextName>, 每个logger都关联到logger上下文,
默认上下文名称为“default”。但可以使用<contextName>设置成其他名字,用于区分不同应用程序的记录。
一旦设置,不能修改。 -->
<contextName>${CONTEXT_NAME}</contextName>
<!-- <appender>是<configuration>的子节点,是负责写日志的组件。
有两个必要属性name和class。
name指定appender名称,
class指定appender的实现类。 -->
<!--控制台日志, 控制台输出 -->
<appender name="stdout" class="ch.qos.logback.core.ConsoleAppender">
<!-- 对日志进行格式化。 -->
<encoder>
<pattern>
<!--格式化输出:%d表示日期,%thread表示线程名,%-5level:级别从左显示5个字符宽度,%msg:日志消息,%n是换行符-->
主机名:%contextName|IP:%ip|timestamp:%d{yyyy-MM-dd HH:mm:ss.SSS}|logLevel:%level|Class:%class|Thred:%thread|Method:%method|Line:%line|Message:%msg%n
</pattern>
</encoder>
</appender>
<!--info文件日志-->
<appender name="infofile" class="ch.qos.logback.core.rolling.RollingFileAppender">
<filter class="ch.qos.logback.classic.filter.LevelFilter">
<!--过滤 INFO-->
<level>INFO</level>
<!--匹配到就禁止-->
<onMatch>ACCEPT</onMatch>
<!--没有匹配到就允许-->
<onMismatch>DENY</onMismatch>
</filter>
<!--rollingPolicy,用来设置日志的滚动策略,当达到条件后会自动将条件前的日志生成一个备份日志文件
,条件后的日志输出到最新的日志文件中。常用的是按照时间来滚动(使用的类TimeBaseRollingPolicy),
还有一种就是基于索引来实现(使用的类FixedWindowRollingPolicy)-->
<rollingPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedRollingPolicy">
<FileNamePattern>${LOG_HOME}/logback-info.%d{yyyyMMdd}.%i.log</FileNamePattern>
<!-- <!–日志文件保留天数–>-->
<!-- 日志文件最大尺寸 -->
<maxFileSize>100MB</maxFileSize>
<!--日志文件保留天数-->
<MaxHistory>30</MaxHistory>
<!-- <!– 窗口索引最小值 –>-->
<!-- <minIndex>1</minIndex>-->
<!-- <!– 窗口索引最大值 –>-->
<!-- <maxIndex>1</maxIndex>-->
</rollingPolicy>
<!-- 激活滚动的条件。 --> <!--日志文件最大的大小-->
<!-- <triggeringPolicy class="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy">-->
<!-- <!– 活动文件的大小,默认值是10MB –>-->
<!-- <maxFileSize>30MB</maxFileSize>-->
<!-- </triggeringPolicy>-->
<!-- 对记录事件进行格式化。 -->
<encoder>
<charset>UTF-8</charset>
<!--格式化输出:%d表示日期,%thread表示线程名,%-5level:级别从左显示5个字符宽度%msg:日志消息,%n是换行符-->
<pattern> 主机名:%contextName|IP:%ip|timestamp:%d{yyyy-MM-dd HH:mm:ss.SSS}|logLevel:%level|Class:%class|Thred:%thread|Method:%method|Line:%line|Message:%msg%n
</pattern>
</encoder>
</appender>
<!--debug日志-->
<appender name="debugfile" class="ch.qos.logback.core.rolling.RollingFileAppender">
<filter class="ch.qos.logback.classic.filter.LevelFilter">
<level>debug</level>
<!--匹配到就禁止-->
<onMatch>ACCEPT</onMatch>
<!--没有匹配到就允许-->
<onMismatch>DENY</onMismatch>
</filter>
<rollingPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedRollingPolicy">
<FileNamePattern>${LOG_HOME}/logback-debug_%d{yyyy-MM-dd}.log</FileNamePattern>
<MaxHistory>30</MaxHistory>
<maxFileSize>50MB</maxFileSize>
</rollingPolicy>
<encoder>
<Pattern>
主机名:%contextName|IP:%ip|timestamp:%d{yyyy-MM-dd HH:mm:ss.SSS}|logLevel:%level|Class:%class|Thred:%thread|Method:%method|Line:%line|Message:%msg%n
</Pattern>
</encoder>
</appender>
<!--error日志-->
<appender name="errorfile" class="ch.qos.logback.core.rolling.RollingFileAppender">
<filter class="ch.qos.logback.classic.filter.LevelFilter">
<level>error</level>
<!--匹配到就禁止-->
<onMatch>ACCEPT</onMatch>
<!--没有匹配到就允许-->
<onMismatch>DENY</onMismatch>
</filter>
<rollingPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedRollingPolicy">
<FileNamePattern>${LOG_HOME}/logback-error_%d{yyyy-MM-dd}.log</FileNamePattern>
<MaxHistory>30</MaxHistory>
<maxFileSize>50MB</maxFileSize>
</rollingPolicy>
<encoder>
<Pattern>
主机名:%contextName|IP:%ip|timestamp:%d{yyyy-MM-dd HH:mm:ss.SSS}|logLevel:%level|Class:%class|Thred:%thread|Method:%method|Line:%line|Message:%msg%n
</Pattern>
</encoder>
</appender>
<!--warn日志-->
<appender name="warnfile" class="ch.qos.logback.core.rolling.RollingFileAppender">
<filter class="ch.qos.logback.classic.filter.LevelFilter">
<level>warn</level>
<!--匹配到就禁止-->
<onMatch>ACCEPT</onMatch>
<!--没有匹配到就允许-->
<onMismatch>DENY</onMismatch>
</filter>
<rollingPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedRollingPolicy">
<FileNamePattern>${LOG_HOME}/logback-warn_%d{yyyy-MM-dd}.log</FileNamePattern>
<MaxHistory>30</MaxHistory>
<maxFileSize>50MB</maxFileSize>
</rollingPolicy>
<encoder>
<Pattern>
主机名:%contextName|IP:%ip|timestamp:%d{yyyy-MM-dd HH:mm:ss.SSS}|logLevel:%level|Class:%class|Thred:%thread|Method:%method|Line:%line|Message:%msg%n
</Pattern>
</encoder>
</appender>
<!-- 特殊的<logger>元素,是根logger。只有一个level属性,应为已经被命名为"root".
level:设置打印级别,大小写无关:TRACE, DEBUG, INFO, WARN, ERROR, ALL 和 OFF,不能设置为INHERITED或者同义词NULL。默认是DEBUG。
<root>可以包含零个或多个<appender-ref>元素,标识这个appender将会添加到这个loger。 -->
<root>
<level value="info,debug,warn,error"/>
<appender-ref ref="stdout"/>
<appender-ref ref="infofile"/>
<appender-ref ref="debugfile"/>
<appender-ref ref="warnfile"/>
<appender-ref ref="errorfile"/>
</root>
<!-- 用来设置某一个 包 或者具体的某一个 类 的日志打印级别、以及指定<appender>,
name:用来指定受此logger约束的某一个包或者具体的某一个类。
level:用来设置打印级别,大小写无关:TRACE, DEBUG, INFO, WARN, ERROR, ALL 和 OFF,还有一个特俗值INHERITED或者同义词NULL,代表强制执行上级的级别。如果未设置此属性,那么当前loger将会继承上级的级别。
additivity:是否向上级logger传递打印信息。默认是true。(这个logger的上级就是上面的root)
<logger>可以包含零个或多个<appender-ref>元素,标识这个appender将会添加到这个logger。-->
<!-- <logger name="zyu.logback.test" level="DEBUG" additivity="true"></logger>-->
</configuration>
done