Logging,Apache Common-Logging/SLF4J, Log4J/Logback

一. JDK 1.4 的Logging

易用性、功能和扩展性很差,可以放弃

日志级别All、FINEST、FINER、FINE、CONFIG、INFO、WARNING、SEVERE、OFF等,级别依次升高。
设置为高级别的情况下,低级别不会输出,比如设置为INFO,则INFO之前的低级别信息将不会输出。
默认有个控制台输出,用于输出INFO级别以上的信息。

 

 

二. commons-logging控件

commons-logging控件的作用是统一JDK Logging与Log4j的API。对于不确定日志方式的系统通常使用commons-logging控件,如spring,hibernate,strus等
如果配置Log4j,commons-logging会把输出原封不动的交给log4j。如果没有log4j,commons-loggin会将相应的输出转化为JDK Loggin输出。
也可以显式的配置。

 

 

三.SLF4J 

简单日记门面(Facade)SLF4J是为各种loging APIs提供一个简单统一的接口,从而使得最终用户能够在部署的时候配置自己希望的loging APIs实现。需要你加入slf4j-jdk14.jar, slf4j-log4j12.jar或logback.jar,将日志调用转发到实际的日志框架。在classpath中有哪个jar包,slf4j就会选择哪个实现。如果错误的同时存在多个jar包,用哪个那就看运气了。

 

使用起来非常简单只要引入如下依赖:

<dependency>
    <groupId>org.slf4j</groupId>
    <artifactId>slf4j-api</artifactId>
</dependency>

<!-- log4j 实际调用slf4j -->
<dependency>
    <groupId>org.slf4j</groupId>
    <artifactId>log4j-over-slf4j</artifactId>
</dependency>

<!-- common-logging 实际调用slf4j -->
<dependency>
    <groupId>org.slf4j</groupId>
    <artifactId>jcl-over-slf4j</artifactId>
</dependency>

<!-- java.util.logging 实际调用slf4j -->
<dependency>
    <groupId>org.slf4j</groupId>
    <artifactId>jul-to-slf4j</artifactId>
</dependency>

 

 

有些第三方的工具包,已经直接使用了log4j, common-logging 或 java.util.logging。如果我们最后决定使用log4j做最终输出,则需要放一个jcl-over-slf4j.jar和 jul-to-slf4j.jar来假装成common-logging和java.util.logging的api,将日志请求转发给slf4j,slf4j再转发给log4j,此时还需要保证,classpath里没有正牌的common-logging.jar。 而原本直接使用log4j的就不需要做任何改动。

slf4j的api不需要写isWarnEnable(),是因为原来common-logging的API,早早就进行了一次字符串拼接,如果不需要打印就白白浪费了时间。而slf4j的代码如下,只在真正需要打印的时候才进行拼接。

logger.info("Hello {}", name);

注意如果参数本身的获取就需要消耗大量的时间,就依然需要用isXXEnable()把代码段圈起来。

   if(logger.isInfoEnabled()){
       logger.info("hello " + userDao.get(id).getName);
   }

 

 

 

四. Log4j

1. 日志级别:ALL、TRACE(跟踪)、DEBUG(调试)、INFO(信息)、WARNING(警告)、ERROR(错误)、FITAL(致命)、OFF 级别依次升高,级别高的会屏蔽级别低的信息。


2. 日志配置
(1)一般配置信息都写在配置文件log4j.properties。启动时会加载classpath下log4j.properties初始化Log4j的输出级别、输出位置、输出信息、输出格式等内容,如果文件不存在,Log4j会打印如下信息:
    log4j:WARN No appenders could be found for logger (com.shaogq.log.Log4jTest).
    log4j:WARN Please initialize the log4j system properly.
    log4j:WARN See http://logging.apache.org/log4j/1.2/faq.html#noconfig for more info.
(2)如果不使用log4j.properties,可用PropertyConfigurator指定配置文件路径
    PropertyConfigurator.configure("conf/common/log4j.properties");
(3)也可以使用Log4j.xml来配置(没见过这么用的)
(4)Spring中使用log4j
    1)在web.xml中添加上下文初始化参数

<context-param>
  <param-name>webAppRootKey</param-name>
  <param-value>path</param-value>
</context-param>

<context-param>
  <param-name>log4jConfigLocation</param-name>
  <param-value>/WEB-INF/log4j.properties</param-value>
</context-param> 

 

    2)添加日志监听器

<listener>
  <listener-class>org.springframework.web.util.Log4jConfigListener</listener-class>
</listener> 

 


    3)添加log4j.properties文件
    获取日志文件输入路径${path}/WEB-INF
    

3.Log4j的执行效率
(1)Log4j内部做了大量的优化、缓存工作,使输出时对服务器的压力、消耗的时间、资源等都达到最小。
(2)Log4j只是在初始化时打开文件,并保持对文件的写控制,直到系统结束才关闭文件,减少I/O次数。
(3)当输出级别低于设定级别时,比如输出级别为Error而使用debug方法时,会直接返回。

4.优化日志代码
即使当前方法级别低于输出级别,仍会对当前方法进行调用并判断。
方法统一为 isXXXEnable() 或 isEnableFor(Priority.XXX)

    if(log.isDebugEnabled()){
        log.debug("...");
    }

 或

    if(log.isEnabledFor(org.apache.log4j.Priority.WARN)){
        log.warn("...");
    }

 


5.日志记录器Logger
public static Logger log = Logger.getLogger(Log4jTtest.class);
(1)Logger.getLogger()方法的参数为Logger的名字
(2)Logger为单例模式,即相同名字的Logger只会有一个实例,如果构建同名Logger,Log4j会返回之前的Logger实例。
(3)命名规则一般以类名为Logger的名称。大小写敏感,用点分开,具有继承关系,log4j.properties通过名称来配置Logger的属性
(4)Log4j中有一个根记录器rootLogger,是所有Logger的父记录器

6.Logger的配置
在log4j.properties配置中,log4j.logger后面配置的是Logger,log4j.appender后面配置的是Appender
    e.g.配置该Logger为DEBUG级别,输出地为A1
    log4j.logger.com.logtest.Log4jTest=DEBUG,A1
(1)如果某个Logger没有配置,则使用它的父亲配置,直到找到为止,一般情况下,只需配置根记录器rootLogger即可,所有的Logger都会沿用rootLogger的配置。
(2)Logger支持多个Appender,用逗号将多个Appender名字隔开即可。

7.rootLogger的配置
(1)根记录器rootLogger直接用log4j.rootLogger配置,rootLogger是所有记录器的父亲,任何记录器都继承rootLogger的配置。
    e.g.配置rootLogger为ERROR级别,输出地为A1
    log4j.rootLogger=ERROR,A1    
(2)如果要对某个Logger进行特殊输出,只需要再配置一下该Logger,覆盖父亲的配置即可,覆盖时,可以只配置级别、输出地,也可两者都配置。
    e.g.配置该logger为DEBUG级别,输出地继承rootLogger配置
    log4j.logger.com.logtest.Log4jConfigTest=DEBUG

8.类别category配置
(1)Logger还有类别(Category)的概念,通过设置类别来配置类别下得所有Logger;
(2)category类似于java中的Package,效果跟Logger的名字等价。
    e.g.作用于类别com.logtest下得所有Logger
    log4j.category.com.logtest=DEBUG

9.输出地Appender
Appender表示日志输出到什么地方,常用的输出地有控制台、文件、数据库、远程服务器等。
Log4j中内置了常用的输出地,一般情况下配置一下即可使用。所有的Appender都实现自org.apache.log4j.Appender接口,在Log4j.properties中,Appender都使用log4j.appender.*配置。

(1)输出到控制台
控制台输出实现类为org.apache.log4j.ConsoleAppender。
    e.g.输出到控制台配置

    #根记录器,ERROR,输出到A1
    log4j.rootLogger=ERROR,A!
    
    #设置com.logtest包下的记录器为DEBUG级别
    log4j.category.com.logtest=DEBUG
    
    #控制台输出
    log4j.appender.A1=org.apache.log4j.ConsoleAppender
    #DEBUG以上级别输出
    #log4j.appender.A1.Threshold=DEBUG
    #编码方式
    #log4j.appender.A1.Encoding=UTF-8
    #是否立即输出
    #log4j.appender.A1.ImmediateFlush=true
    #使用System.err输出
    #log4j.appender.A1.Target=System.err
    #输出格式,表达式配置
    log4j.appender.A1.layout=org.apache.log4j.PatternLayout
    log4j.appender.A1.layout.ConversionPattern=%-d{yyyy-MM-dd HH:mm:ss,SSS} [%C]-[%p] %m%n

 


    
1)控制台输出需要配置layout属性,最常用的是正则表达式格式。
2)控制台输出的信息一般是TRACE、DEBUG或INFO级的,只在开发、调试时才启用。
3)被注释掉的属性都是可选属性,Encoding设置编码方式,ImmediateFlush设置是否缓存,Target设置输出到System.out还是System.err。
4)Threshold用来设置该Appender的级别,只对本Appender生效。所有的Appender都可以通过设置Threshold来设置本Appender的启用级别。

(2)输出到文件
文件输出(FileAppender)把日志输出到指定文件。文件输出的实现类org.apache.log4j.FileAppender,配置时需要用File指定文件名称。可以使用相对路径,也可以使用绝对路径。
    e.g.输出到文件

    log4j.rootLogger=ERROR, f,stdout
    log4j.logger.com.logtest.Log4jConfigTest=debug
    
    #输出到文件
    log4j.appender.f=org.apache.log4j.FileAppender
    #文件位置
    log4j.appender.f.File=../log/tomcat.log
    #追加文件内容
    log4j.appender.f.Append=true
    #输出格式的表达式
    log4j.appender.f.layout=org.apache.log4j.PatternLayout
    log4j.appender.f.layout.ConversionPattern=%-d{yyyy-MM-dd HH:mm:ss,SSS} [%C]-[%p] %m%n

 

1)可选参数Append配置是否在原文件内容上追加日志。如果为false,Logger初始化时会清掉文件内容,也就是所每次重启程序,原来的日志会丢失。如果为true,日志文件文件会越来越大。
2)web下文件位置说明,tomcat.log输出到tomcat/bin目录下;/tomcat.log输出到根目录下;../tomcat.log输出到tomcat目录下;推荐输出地址../logs/XXX.log

(3)输出到按大小滚动文件
按大小滚动文件输出(RollingFileAppender)把日志输出到指定的文件,文件达到指定的大小时,会自动更名。
按尺寸滚动文件输出类为org.apache.log4j.RollingFileAppender,需配置文件名称、文件的最大尺寸。
    e.g.输出到按大小滚动文件

    #DEBUG级别,两个输出,文件与滚动文件
    log4j.logger.com.logtest.Log4jConfigTest=DEBUG,f,rolling_file
    
    #输出到文件
    log4j.appender.f=org.apache.log4j.FileAppender
    log4j.appender.f.File=../log/tomcat.log
    log4j.appender.f.Append=true
    log4j.appender.f.layout=org.apache.log4j.PatternLayout
    log4j.appender.f.layout.ConversionPattern=%-d{yyyy-MM-dd HH:mm:ss,SSS} [%C]-[%p] %m%n
    
    #输出到滚动文件
    log4j.appender.rolling_file=org.apache.log4j.RollingFileAppender
    #DEBUG以上级别输出
    log4j.appender.rolling_file.Threshold=DEBUG
    #滚动文件位置
    log4j.appender.rolling_file.File=../log/rolling.log
    #追加方式
    log4j.appender.rolling_file.Append=true
    #文件达到10K就自动更名
    log4j.appender.rolling_file.MaxBackupIndex=100
    log4j.appender.f.layout=org.apache.log4j.PatternLayout
    log4j.appender.f.layout.ConversionPattern=%-d{yyyy-MM-dd HH:mm:ss,SSS} [%C]-[%p] %m%n

 配置的滚动文件名为rolling.log,文件最大为10K,当rolling.log达到10K时,就会自动更名为rolling.log1、rolling.log2 ......直到rolling.log100。


(4)输出到按日期滚动文件
输出日期滚动文件输出类为org.apache.log4j.DailyRollingFileAppender    
    e.g.输出到按日期滚动文件

    log4j.logger.com.logtest.Log4jConfigTest=DEBUG,daily_rolling
    
    #滚动文件
    log4j.appender.daily_rolling=org.apache.log4j.DailyRollingFileAppender
    #滚动文件位置
    log4j.appender.daily_rolling.File=../log/daily_rolling.log
    #滚动日期格式
    log4j.appender.daily_rolling.DatePattern=.yyyy-MM-dd
    #输出格式表达式
    log4j.appender.f.layout=org.apache.log4j.PatternLayout
    log4j.appender.f.layout.ConversionPattern=%-d{yyyy-MM-dd HH:mm:ss,SSS} [%C]-[%p] %m%n

 


    
日志文件名为daily_rolling.log,日期格式为yyyy-MM-dd,进入新的一天后文件会被自动更名,格式为daily_rolling.log.2012-6-18


(5)输出到JDBC数据库
数据库输出(JDBCAppender)通过JDBC连接把日志输出到数据库中,输出类为org.apache.log4j.JDBCAppender
配置:JDBC驱动、连接字符串、用户名、密码、SQL语句;
注意:JDBCAppender需要相应的数据库驱动。先创建数据库,并建立数据库日志表。
    e.g.输出到MySql数据库

    #init.sql
    CREATE DATABASE log4j CHARACTER SET utf8;
    USE log4j;
    CREATE TABLE tb_log{
    id INT AUTO_INCREMENT,
    date VARCHAR(255),
    priority VARCHAR(255),
    message TEXT,
    classname VARCHAR(255),
    PRIMARY KEY(id)
    };

 

    #log4j.properties

    log4j.logger.com.logtest.Log4jConfigTest=DEBUG,DATABASE
    
    #输出到数据库
    log4j.appender.DATABASE=org.apache.log4j.JDBCAppender
    #ERROR以上输出
    log4j.appender.DATABASE.Threshold=ERROR
    #数据库连接URL
    log4j.appender.DATABASE.URL=jdbc:mysql://localhost:3306/log4j
    #数据库驱动
    log4j.appender.DATABASE.driver=com.mysql.jdbc.Driver
    #用户名
    log4j.appender.DATABASE.user=root
    #密码
    log4j.appender.DATABASE.password=12345678
    #执行SQL的语句,列内容为表达式
    log4j.appender.DATABASE.sql=INSERT INTO tb_log(date
    , priority, message, classname) VALUES('%d', '%p', '%m', '%c')
    #Oracle的例子
    #log4j.appender.JDBC.sql=INSERT INTO oracle_log4j
    (id, type, username, user_id, create_date, thread, priority
    , category, message) values(rz_log4j_seq.nextval ,'{m}'
    ,'{m}',{m},to_date('%d{yyyy-MM-dd HH:mm:ss}'
    ,'yyyy-MM-dd hh24:mi:ss'), '%t', '%-5p', '%c', '{m}') && %m

 


(6)输出到SOCKET套接字
套接字输出(SocketAppender)将日志通过网络TCP协议发送给远程服务器,SocketAppender会与远程服务器建立Socket连接,
将日志信息封装为LoggingEvent对象,串行化(Serialize)后发送给对方,输出类为org.apache.log4j.net.SocketAppender
    

(7)输出到SMTP邮件
邮件输出(SMTPAppender)将日志以邮件的形式发送出去,输出类为org.apache.log4j.net.SMTPAppender


10.日志格式化器Layout
(1)PatternLayout布局
                Log4j的常用参数

参数 描述 示例
c 输出Logger所在的类别,允许%c{数字}输出部分名称(从右往左数)

%c输出a.b.clazz

%c{1}输出clazz

%c{2}输出b.clazz

%c{3}输出a.b.clazz

C 输出Logger所在类的名称,有时候Logger的名称不同于类名,允许%C{数字}

%C输出a.b.clazz

%C{1}输出clazz

%C{2}输出b.clazz

%C{3}输出a.b.clazz

d 输出日期,允许使用 %d{yyyy-MM-dd HH-mm-ss} 格式化日期,支持log4j自己的日期格式,ABSOLUTE、DATE、ISO8601等

%d 输出2012-6-18 16:03:49,353

%d{yyyy-MM-dd} 输出2012-6-18

%d{ABSOLUTE} 输出16:03:49,353

%d{DATE} 输出 18 六月 2012 16:03:49,353

F 输出所在类文件名称 %F 输出 Log4jConfigTest.java
l 输出语句所在行数,包括 类名、方法名、文件名、行数 %l 输出 com.logtest.Log4jConfigTest(Log4jConfigTest.java:34)
L 只输出语句所在行数 %L 输出 34
m 输出日志 输出日志,即log.info("")、log.debug("")参数
M 输出方法名 %M 输出Test
n 换行,win下\r\n,linux下\n 换行
p 输出日志级别(priority) DEBUG、INFO、ERROR、FITAL等
r 输出程序启动到输出时间间隔 %r 输出3000
t 输出当前线程的名称 %t 输出main、Thread-0、Thread-1等
% %%用来输出百分号  

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

Log4j允许设置输出内容的长度等,不够长会用空格补齐,
设置方法是在%与参数符号间添加数字,负数表示左对齐,数字表示最小宽度,不足时用空格补齐。
可以设置最大宽度,如果超出则截取,方法是用小数点+数字设置,例如 %.30p

            长度调整

格式 对齐方式 最小宽度 最大宽度 描述 示例
[%10p] 右对齐 10   正数右对齐,最小宽度 [      ERROR]
[%-10p] 左对齐 10   负数左对齐,最小宽度 [ERROR      ]
[%.4p]     4 最大宽度 [RROR]
[%10.20p] 右对齐 10 20 正数右对齐,最大最小宽度 [      ERROR]
[%-10.20p] 左对齐 10 20 正数右对齐,最大最小宽度 [      ERROR]

 

 

(2)HTMLLayout布局
HTMLLayout将日志格式转化为HTML代码,输出到文件后,可以直接用浏览器浏览
使用HTMLLayout时,文件后缀一般为.html

    log4j.rootLogger=DEBUG,f
    log4j.appender.f=org.apache.log4j.FileAppender
    log4j.appender.f.File=../log/log.html
    log4j.appender.f.Append=false
    #输出格式的表达式
    log4j.appender.f.layout=org.apache.log4j.HTMLLayout

 



(3)XMLLayout布局
XMLLayout把日志内容格式化为XML文件

    log4j.rootLogger=DEBUG,f
    log4j.appender.f=org.apache.log4j.FileAppender
    log4j.appender.f.File=../log/log.log
    log4j.appender.f.Append=false
    #输出格式的表达式
    log4j.appender.f.layout=org.apache.log4j.XMLLayout

 


注意:XMLLayout生成的并不是完整的XML文件,而只是XML文件中的一部分,因此无法直接打开、解析。这个片段包括系列的<log4j:event>标签

 


五.LOGBack 


Logback是由log4j创始人设计的又一个开源日记组件。logback当前分成三个模块:logback-core,logback- classic和logback-access。logback-core是其它两个模块的基础模块。logback-classic是log4j的一个 改良版本。此外logback-classic完整实现SLF4J API使你可以很方便地更换成其它日记系统如log4j或JDK14 Logging。logback-access访问模块与Servlet容器集成提供通过Http来访问日记的功能。

一份可滚动结果集的配置文件

logback.xml

 

<?xml version="1.0" encoding="UTF-8"?>
<configuration>
    <jmxConfigurator />

    <appender name="console" class="ch.qos.logback.core.ConsoleAppender">
        <encoder>
            <pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
        </encoder>
    </appender>

    <appender name="rollingFile" class="ch.qos.logback.core.rolling.RollingFileAppender">
        <append>true</append>
        <Encoding>UTF-8</Encoding>
        <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
            <FileNamePattern>../logs/projectname.%d{yyyy-MM-dd}[%i].log</FileNamePattern>
            <MaxHistory>10</MaxHistory>
            <TimeBasedFileNamingAndTriggeringPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedFNATP">
                <MaxFileSize>5MB</MaxFileSize>
            </TimeBasedFileNamingAndTriggeringPolicy>
        </rollingPolicy>
        <layout class="ch.qos.logback.classic.PatternLayout">
            <pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
        </layout>
    </appender>

    <!--log4jdbc -->
<logger name="jdbc.sqltiming" level="INFO"/>

    <logger name="com.projectname" level="INFO" />

    <root level="WARN">
        <appender-ref ref="console" />
        <appender-ref ref="rollingFile" />
    </root>
</configuration>
 

 

 

 

六.关系

Slf4j和Commons-Logging都是简单日记门面(Facade),但SLF4J经成为Logger的事实标准API。

java.util.logging, log4j ,logback,log4j2 一个比一个强。其中logback是log4j作者觉得log4j已经太烂不想再改了,重新写的一个实现。Log4j本来一统江湖好好的,后来被人说方法上太多同步修饰符,在高并发下性能太烂。Netflix的blitz4j就重新实现了一次log4j项目,去掉了大量的同步修饰符,不过其负责人自己说,新项目还是建议直接用logback,所以SpringSide就用了logback。

 

不过,后来apache社区又说,slf4j和logback都是log4j作者开的qos.ch公司的产品,日志是件很重要的事情,不应该操控在一家公司手里。所以又以纯社区驱动搞了log4j2,参考了logback,也做了一些自己的改动。不过现在还是漫长的beta版。

 

 

 

你可能感兴趣的:(logback)