博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Logging,Apache Common-Logging/SLF4J, Log4J/Logback
阅读量:4048 次
发布时间:2019-05-25

本文共 12102 字,大约阅读时间需要 40 分钟。

一. 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包,用哪个那就看运气了。

 

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

org.slf4j
slf4j-api
org.slf4j
log4j-over-slf4j
org.slf4j
jcl-over-slf4j
org.slf4j
jul-to-slf4j

 

 

有些第三方的工具包,已经直接使用了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中添加上下文初始化参数

webAppRootKey
path
log4jConfigLocation
/WEB-INF/log4j.properties

 

    2)添加日志监听器

org.springframework.web.util.Log4jConfigListener

 

    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

 

%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n
true
UTF-8
../logs/projectname.%d{yyyy-MM-dd}[%i].log
10
5MB
%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n
 

 

 

 

六.关系

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版。

 

 

 

转载地址:http://yoyci.baihongyu.com/

你可能感兴趣的文章
vc学习之键盘事件OnKeyDown
查看>>
近期工作
查看>>
春寒料峭,原来春天这样走近
查看>>
电脑坏了--关于联想笔记本声卡驱动
查看>>
C#中枚举类型在switch语句中值对照问题
查看>>
GridView中根据特殊标记设置不可编辑
查看>>
PowerDesinger中生成数据库时将域的内容转化为实际的字段
查看>>
通过模板将GridView导出为Excel
查看>>
迭代开发过程及一些原则
查看>>
Windows IIS服务器CA认证安装
查看>>
Asp.net 中Excel通过模板导出中发布问题
查看>>
64位机器配置CA认证一些小问题
查看>>
SVN自动完全备份
查看>>
开发中的Warning原来也很有用
查看>>
Silverlight Toolkit例子代码中缺少System.Windows.Controls.Samples.Common.dll的解决办法
查看>>
项目经理培训后的一点感概
查看>>
vc学习之窗口大小发生变更时使控件自动摆放到合适的位置
查看>>
Oracle 获取系统日期时间,导出数据库
查看>>
黄巢的菊花,非常喜欢,贴出来共享一下
查看>>
在光纤环网中的b/s与c/s的比对
查看>>