博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
【译】log4j介绍
阅读量:6037 次
发布时间:2019-06-20

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

本文是对log4j官网Introduction部分的翻译,原文链接地址:。

Introduction:

几乎每个大型应用都包含自己的日志API。1996年,为了整个项目的一致性,E.U. SEMPER项目团队决定开发自己的日志API。经过无数次的改进,这款日志API成为了Java领域非常流行的日志package,这就是log4j。

在代码中加日志进行调试是一种低级的方式。但因为调试工具并不总是可以使用,所以打日志有时候是唯一的调试方法。例如多线程应用和分布式应用。
有经验表明在软件开发周期中,日志组件占有重要的地位。加入日志有许多好处。可以通过日志精确了解应用运行的状态。加入代码中的日志,无需人们手工干预就可自动生成输出结果。日志的输出结果可以保存在永久存储介质上,方便今后对其进行查看。此外,在软件开发周期中,充足丰富的日志也可以当做审计材料使用。
日志也有缺点,它会使应用程序变慢。如果日志过多,还会导致屏幕闪动。为了缓和这些缺点,log4j被设计成可靠的、快速的和可扩展的。由于日志很少是一个应用中关注的重点,所以log4j的API尽可能设计的简单易懂。

Loggers, Appenders and Layouts

log4j主要有三大组件——loggers、appenders和layouts。这三大组建共同协作,使开发者可以根据不同的日志级别和日志类型输出信息,并且可以指定信息输出的格式和信息输出的目的地。

Logger hierarchy

相比于使用简单的System.out.println语句,日志API最大的优势就是它可以禁止某一类型的日志输出,同时又不影响其它类型的日志输出。要实现这种能力,需要开发人员根据某种条件,将日志划分为不同的类型。老版本的log4j将Category类作为核心就是由于上面这个原因。但log4j到1.2版本时,已经使用Logger类代替了原来的Category类。对于那些熟悉老版本log4j的人,可以简单把Logger类当做Category类的一个别名。

Loggers是被命名的实体,Logger的名称是大小写敏感的,并且遵循层次命名规则。
举个例子,命名为“com.foo”的logger是命名为“com.foo.Bar”的logger的父亲。命名为“java”的logger是命名为“java.util”的logger的父亲,是命名为“java.util.Vector”的祖先。这种命名规则应该被许多研发人员所熟悉。
root logger位于整个logger继承体系的最顶端,相比于普通logger它有两个特别之处:

  1. root logger总是存在。
  2. root logger不能通过名称获取。

可以通过调用Logger类的静态方法getRootLogger获取root logger对象。其它普通logger的实例可以通过Logger类的另一个静态方法getLogger获取。getLogger方法接受一个参数作为logger的名字。

Logger类中的其它一些基本方法如下所示:

package org.apache.log4j;  public class Logger {    // Creation & retrieval methods:    public static Logger getRootLogger();    public static Logger getLogger(String name);    // printing methods:    public void trace(Object message);    public void debug(Object message);    public void info(Object message);    public void warn(Object message);    public void error(Object message);    public void fatal(Object message);    // generic printing method:    public void log(Level l, Object message);}

Loggers可以被分配日志级别,可以分配的级别如下:

这些级别被定义在org.apache.log4j.Level类中。虽然你也可以通继承Level类定义你自己的专有级别,但是我们不鼓励你这样做。
如果一个logger没有指定任何level,那么这个logger会从它的父亲那里继承level。
为了保证所有的logger可以最终被指定一个level,root logger总是被分配一个level。
下面四个表是上面规则的例子:
Example 1
图片描述

在第一个例子中,只有root logger被分配了一个level值Proot,Proot会被其它所有的logger——x、x.y、x.y.z继承。

Example 2
图片描述

在第二个例子中,所有的logger都被分配了一个level值,就不需要继承level了。

Example 3
图片描述

在第三个例子中,root、x和x.y.z三个logger分别被分配了Proot、Px和Pxyz三个level值,x.y这个logger从它的父亲那里继承level值。

Example 4
图片描述

在第四个例子中,root和x两个logger分别被分配了Proot和Px这两个level值。x.y和x.y.z两个logger则从离自己最近的祖先x继承level值。

可以调用logger实例的printing方法输入日志。printing方法包括debug、info、warn、error、fatal和log。
按照定义,printing方法决定了日志请求的等级。例如,c是一个logger实例,c.info("..")语句请求输出INFO级别的日志。
只有日志请求级别大于等于日志级别的时候,日志请求才会被准许输出信息。否则,日志请求会被禁止。这条规则是log4j的核心。日志的level是有序的。对于标准的日志级别:DEBUG<INFO<WARN<ERROR<FATAL。
下面是这条规则的一个例子:

// get a logger instance named "com.foo"   Logger  logger = Logger.getLogger("com.foo");   // Now set its level. Normally you do not need to set the   // level of a logger programmatically. This is usually done   // in configuration files.   logger.setLevel(Level.INFO);   Logger barlogger = Logger.getLogger("com.foo.Bar");   // This request is enabled, because WARN >= INFO.   logger.warn("Low fuel level.");   // This request is disabled, because DEBUG < INFO.   logger.debug("Starting search for nearest gas station.");   // The logger instance barlogger, named "com.foo.Bar",   // will inherit its level from the logger named   // "com.foo" Thus, the following request is enabled   // because INFO >= INFO.   barlogger.info("Located nearest gas station.");   // This request is disabled, because DEBUG < INFO.   barlogger.debug("Exiting gas station search");

用相同的名字参数调用getLogger方法总是会返回同一个logger对象的引用,例如在下面两行代码中:

Logger x = Logger.getLogger("wombat");   Logger y = Logger.getLogger("wombat");

x和y引用的是同一个logger对象。

这样在配置好一个logger实例之后,可以很方便的在代码的其它地方获取到这个logger实例,而无需传递logger实例的引用。
与生物学中的父子关系不同,log4j中的父亲不一定总是早于它的孩子出生。log4j中的logger实例可以以任意的顺序被构造或配置。一个parent logger即使在它的后代之后被实例化,它们也依然可以建立起父子关系。
log4j的配置通常会在应用初始化时被完成。最常用的方式是通过读取配置文件完成配置。稍后会讨论这个过程。
在log4j中对一个logger实例命名非常的简单,在每一个类中可以有一个静态的logger实例对象,可以用类的完全限定名作为logger实例的名字。这对于定义一个logger非常有用,由于在输出日志的时候可以带有logger实例的名字,所以这种用类的完全限定名作为logger实例的名字可以很容易看出日志发生的位置。当然这只是一种通用做法,log4j对此并没有限制,开发人员可以随意指定logger实例的名字。但不管怎样,用类的完全限定名作为logger实例的名字是一个非常好的方式。

Appenders and Layouts

禁止和允许日志输出的能力只是全部功能的一部分。log4j允许将日志输出到多个目的地。在log4j的术语中,日志输出目的地被称为appender。目前,存在的appender包括命令行、文件、GUI组件、远程socket服务器、JMS、NT事件日志和远程UNIX Syslog后台进程。并且可以支持异步的方式记录日志。

一个logger实例可以同时挂载多个appender。
addAppender方法向logger实例添加一个appender。对于每一个被允许的日志输出请求,logger实例不仅将该请求转发到自己所有的appender上,而且还将日志输出请求转发到它祖先上的所有appender上。例如,如果一个命令行appender被添加到root logger上,那么所有被允许的日志请求至少会输出到命令行中。如果在这个基础上再向C logger中添加一个文件appender,那么对于C和它的后代,被允许的日志会同时输出到命令行和文件。通过将additivity flag设置为false,可以覆盖这种默认行为。
Appender Additivity:对C logger的日志输出请求会转发到C自己和它祖先们的全部appender。这种行为用术语“appender additivity”表示。如果 P是C的祖先,P将additivity flag设置为false。那么C的日志会输出到它自己的appender和C到P之间(包括P)每个logger的appender,而不会输出到P以上祖先的appender。对于每个logger,它的additivity flag默认是设置为true的。
下面的表格是这样的一个例子:
图片描述
通常,研发不仅希望指定日志输出的目的地,而且希望能够指定日志输出的格式。可以在appender上关联一个layout用于指定日志输出格式。layout会按照用户的意愿输出一定格式的日志信息。
PatternLayout可以让用户像使用C语言中的printf那样使用格式化表达式定制日志输出的格式。
例如,使用PatternLayout的表达式"%r [%t] %-5p %c - %m%n”可以包含下面日志信息:
176 [main] INFO org.foo.Bar - Located nearest gas station.
第一个字段是程序启动到现在经过的毫秒数,第二个字段是输出日志的线程,第三个字段是日志的级别,第四个字段是logger的名字。’-’后面的文本是日志输出信息。%n是换行。
log4j会按照用户指定的具体条件输出日志内容。例如,如果你频繁的需要输出Orange类对象的日志,那么一可以注册一个OrangeRenderer,每当输出orange的日志时,它都会被调用。
Object rendering follows the class hierarchy. For example, assuming oranges are fruits, if you register a FruitRenderer, all fruits including oranges will be rendered by the FruitRenderer, unless of course you registered an orange specific OrangeRenderer.
Object renderers have to implement the ObjectRenderer interface.

Configuration

在现有应用中加入日志需要大量的工作。调研表明,大约有接近4%的代码跟日志有关。因此,即使不是那么大的应用也会有成千上万行的日志代码。Given their number, it becomes imperative to manage these log statements without the need to modify them manually.

og4j环境是完全可以通过写程序进行配置的。然而,使用配置文件对log4j进行配置会更加的灵活。目前,配置文件可以采用XML和properties两种格式的文件。

import com.foo.Bar; // Import log4j classes. import org.apache.log4j.Logger; import org.apache.log4j.BasicConfigurator; public class MyApp {   // Define a static logger variable so that it references the   // Logger instance named "MyApp".   static Logger logger = Logger.getLogger(MyApp.class);   public static void main(String[] args) {     // Set up a simple configuration that logs on the console.     BasicConfigurator.configure();     logger.info("Entering application.");     Bar bar = new Bar();     bar.doIt();     logger.info("Exiting application.");   } }

MyApp在导入了log4j相关的类,然后用MyApp的全限定类名定义了一个静态的logger实例变量。

MyApp中使用的Bar类:

package com.foo; import org.apache.log4j.Logger; public class Bar {   static Logger logger = Logger.getLogger(Bar.class);   public void doIt() {     logger.debug("Did it again!");   } }

调用BasicConfigurator.configure方法创建了一个非常简单的log4j配置。它以硬编码的方式向root logger中添加了一个ConsoleAppender,日志输出会使用PatternLayout的模板"%-4r [%t] %-5p %c %x - %m%n”进行格式化。注意默认情况下,root logger被分配的日志级别是Level.DEBUG。

上面程序输入的日志为:

0    [main] INFO  MyApp  - Entering application.36   [main] DEBUG com.foo.Bar  - Did it again!51   [main] INFO  MyApp  - Exiting application.

下面的图形是MyApp在调用完BasicConfigurator.configure方法之后的对象图:

图片描述
前面的这种方式只能一直输出同一种配置模式的日志信息,可以很容易的在MyApp启动时修改日志配置信息,使其输出不同配置模式的日志。

import com.foo.Bar; import org.apache.log4j.Logger; import org.apache.log4j.PropertyConfigurator; public class MyApp {   static Logger logger = Logger.getLogger(MyApp.class.getName());   public static void main(String[] args) {     // BasicConfigurator replaced with PropertyConfigurator.     PropertyConfigurator.configure(args[0]);     logger.info("Entering application.");     Bar bar = new Bar();     bar.doIt();     logger.info("Exiting application.");   } }

这个版本的MyApp使用PropertyConfigurator解析文件并对日志进行配置。

下面这个配置文件产生的配置结果与之前使用BasicConfigurator产生的结果完全相同。

# Set root logger level to DEBUG and its only appender to A1.log4j.rootLogger=DEBUG, A1# A1 is set to be a ConsoleAppender.log4j.appender.A1=org.apache.log4j.ConsoleAppender# A1 uses PatternLayout.log4j.appender.A1.layout=org.apache.log4j.PatternLayoutlog4j.appender.A1.layout.ConversionPattern=%-4r [%t] %-5p %c %x - %m%n

如果我们不再对com.foo包中任何组件输出的日志感兴趣,下面的配置文件可以实现这一点:

log4j.rootLogger=DEBUG, A1log4j.appender.A1=org.apache.log4j.ConsoleAppenderlog4j.appender.A1.layout=org.apache.log4j.PatternLayout# Print the date in ISO 8601 formatlog4j.appender.A1.layout.ConversionPattern=%d [%t] %-5p %c - %m%n# Print only messages of level WARN or above in the package com.foo.log4j.logger.com.foo=WARN

现在MyApp的日志输出如下:

2000-09-07 14:07:41,508 [main] INFO  MyApp - Entering application.2000-09-07 14:07:41,529 [main] INFO  MyApp - Exiting application.

由于logger com.foo.Bar没有被指定任何的日志级别,所以它会从com.foo上继承,在配置文件中指定com.foo的日志级别是WARN,而代码Bar.doIt中的log语句日志请求输出的是DEBUG级别,要比WARN级别低,所以doIt方法中的日志请求不会被响应。

下面是另一个配置文件,它使用了多个appenders:

log4j.rootLogger=debug, stdout, Rlog4j.appender.stdout=org.apache.log4j.ConsoleAppenderlog4j.appender.stdout.layout=org.apache.log4j.PatternLayout# Pattern to output the caller's file name and line number.log4j.appender.stdout.layout.ConversionPattern=%5p [%t] (%F:%L) - %m%nlog4j.appender.R=org.apache.log4j.RollingFileAppenderlog4j.appender.R.File=example.loglog4j.appender.R.MaxFileSize=100KB# Keep one backup filelog4j.appender.R.MaxBackupIndex=1log4j.appender.R.layout=org.apache.log4j.PatternLayoutlog4j.appender.R.layout.ConversionPattern=%p %t %c - %m%n

使用这个配置文件会向命令行输出如下日志信息:

INFO [main] (MyApp2.java:12) - Entering application.DEBUG [main] (Bar.java:8) - Doing it again!INFO [main] (MyApp2.java:15) - Exiting application.

此外,root logger被分配的第二个appender,是将日志信息直接输出到example.log文件中的。当example.log文件到达100KB时,会发生roll-over。这时老版本的example.log会自动移动到example.log.1。

注意到要改变日志行为无需重新编译代码。我们可以改变配置文件让其日志输出到UNIX Syslog daemon、将所有的com.foo输出的日志都重定向到NT Event logger、或者将日志事件转发到远程的log4j服务器。

Default Initialization Procedure

log4j库没有对它的运行环境做过任何的假设。也就是说,log4j没有任何默认的appender。然而在一些环境下,日志类的静态初始化器会自动尝试配置log4j。Java从语言层面保证在类加载的时候,静态初始化器会被调用一次且仅被调用一次。但是要额外留意不同的classloader可能会对同一个类加载多个副本。

The default initialization is very useful in environments where the exact entry point to the application depends on the runtime environment. For example, the same application can be used as a stand-alone application, as an applet, or as a servlet under the control of a web-server.
The exact default initialization algorithm is defined as follows:

  1. 设置log4j.defaultInitOverride系统属性为非false值会导致log4j略过默认的初始化过程
  2. 设置log4j.configuration系统属性的字符串值。通过设置log4j.configuration系统属性的值指定默认初始化文件是最常用的方式。如果系统属性log4j.configuration没有被明确定义,则为它分配默认值为log4j.properties。log4j.configuration的值记为resource变量。
  3. 尝试将resource变量转换为url。
  4. 如果resource变量无法转化为url,比如在转化url时发生MalformedURLException异常,那么调用org.apache.log4j.helpers.Loader.getResource(resource, Logger.class)在classpath中寻找resource并返回一个url。
  5. 如果找不到url,终止默认初始化,否则使用url对log4j进行配置。

如果url不是以“.xml”为后缀,那么将会使用PropertyConfigurator解析url并对log4j进行配置。如果url的后缀是“.xml”,那么将会使用DOMConfigurator完成上述工作。你可以随意指定一个定制的配置器(configurator)。log4j.configuratorClass系统属性的值就是你定制配置器的全限定类名。你自己定制的配置器必须实现Configurator这个接口。

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

你可能感兴趣的文章
linux上架设l2tp+ipsec ***服务器
查看>>
curl指令的使用
查看>>
LNAMP第二版(nginx 1.2.0+apache 2.4.2+php 5.4)
查看>>
基于用户投票的排名算法(二):Reddit
查看>>
css3中变形与动画(一)
查看>>
[实战]MVC5+EF6+MySql企业网盘实战(23)——文档列表
查看>>
[译] ES2018(ES9)的新特性
查看>>
正则与sed,grep,awk三剑客
查看>>
诊断一句SQL不走索引的原因
查看>>
Linux pipe函数
查看>>
(原創) 如何設計一個數位相框? (SOC) (Quartus II) (SOPC Builder) (Nios II) (TRDB-LTM) (DE2-70)...
查看>>
/etc/profile文件内容
查看>>
一页纸IT项目管理:大道至简的实用管理沟通工具
查看>>
汽车知识:车内异味的清除方法
查看>>
IE6 7下绝对定位引发浮动元素神秘消失
查看>>
浏览器的回流和重绘及其优化方式
查看>>
2.4 salt grains与pillar jinja的模板
查看>>
VDI序曲二十 桌面虚拟化和RemoteApp集成到SharePoint 2010里
查看>>
移动互联网,入口生死战
查看>>
JAVA多线程深度解析
查看>>