设为首页 收藏本站
查看: 526|回复: 0

[经验分享] Logging,Apache Common-Logging/SLF4J, Log4J/Logback

[复制链接]

尚未签到

发表于 2017-1-6 09:04:59 | 显示全部楼层 |阅读模式
一. 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]右对齐1020正数右对齐,最大最小宽度[      ERROR]
[%-10.20p]左对齐1020正数右对齐,最大最小宽度[      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版。

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-324497-1-1.html 上篇帖子: Apache Rewrite 规则的常见应用 下篇帖子: 异常:org.apache.struts2.json.JSONException: org.apache.struts2.json.JSONExcepti
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表