log4j框架logger的继承关系以及使用场景

log4j日志框架logger是存在继承关系的,我们一般都会在log4j.properties文件中定义log4j.rootLogger。其他所有logger都继承自这个rootLooger。考虑下面这种场景:假如我们有2个类HelloLog4j和A。

  1. package aty.log;
  2. import org.apache.log4j.Logger;
  3. import aty.log.service.A;
  4. public class HelloLog4j {
  5. private static Logger logger = Logger.getLogger(HelloLog4j.class);
  6. public static void main(String[] args) {
  7. logger.debug("log in main.");
  8. new A().run();
  9. }
  10. }
  1. package aty.log.service;
  2. import org.apache.log4j.Logger;
  3. public class A {
  4. private static Logger logger = Logger.getLogger(A.class);
  5. public void run() {
  6. logger.error("log in A.java");
  7. }
  8. }

log4j.properties文件配置如下:

  1. #rootLogger can print DEBUG to console
  2. log4j.rootLogger=debug, console
  3. log4j.appender.console=org.apache.log4j.ConsoleAppender
  4. log4j.appender.console.Threshold=debug
  5. log4j.appender.console.ImmediateFlush=true
  6. log4j.appender.console.Target=System.err
  7. log4j.appender.console.layout=org.apache.log4j.PatternLayout

运行HelloLog4j.java控制台日志如下:可以看到这2个类使用的logger都是rootLogger。



现在我们有这个需求:让A.java中日志的打印到单独的a.log日志文件中,而HelloLog4j中的日志还是打印到控制台。也就是说:我们想让A和HelloLog4j这2个类使用不同的logger。java代码不用修改,我们将log4j.properties修改如下:

  1. #rootLogger can print DEBUG to console
  2. log4j.rootLogger=debug, console
  3. log4j.appender.console=org.apache.log4j.ConsoleAppender
  4. log4j.appender.console.Threshold=debug
  5. log4j.appender.console.ImmediateFlush=true
  6. log4j.appender.console.Target=System.err
  7. log4j.appender.console.layout=org.apache.log4j.PatternLayout
  8. log4j.appender.console.layout.ConversionPattern=[%d{yyyy-MM-dd HH:mm:ss,SSS}]-[%t]-[%X{ip}]-[%l]-[%p] %m%n
  9. #A.java is special.we use a individual appender for it.
  10. #A class's full qulified qualified name is logger's name.
  11. log4j.logger.aty.log.service.A=DEBUG, testA
  12. log4j.appender.testA=org.apache.log4j.FileAppender
  13. log4j.appender.testA.Threshold=warn
  14. log4j.appender.testA.ImmediateFlush=true
  15. log4j.appender.testA.Append=true
  16. log4j.appender.testA.File=c:/a.log
  17. log4j.appender.testA.layout=org.apache.log4j.PatternLayout
  18. log4j.appender.testA.layout.ConversionPattern=[%d{yyyy-MM-dd HH:mm:ss,SSS}]-[%t]-[%X{ip}]-[%l]-[%p] %m%n
  19. #avoid print to parent logger.
  20. log4j.additivity.aty.log.service.A= false

再次运行HelloLog4j可以看到:A.java中的日志并没有打印到控制台上,而是打印到了a.log文件中。



我们来解释下上面的配置:

log4j.logger.aty.log.service.A=DEBUG, testA    我们用A的全类名定义了一个logger。

log4j.additivity.aty.log.service.A= false     避免日志打印到rootLogger中。如果这里设置成true,那么A.java中的日志既会打印到控制台上,也会打印到a.log文件中。

现在我们可以介绍下logger的继承关系了,比如我们上面的类aty.log.service.A中使用了logger来写日志。那么log4j会先查找名称是"aty.log.service.A"的logger,如果没有找到,向上查找名称是"aty.log.service"的logger,如果还没有找到那么继续向上查找,查找的最顶层就是rootLogger。这就是log4j中logger的继承关系。rootLogger一定要在配置,其他特定类或者特定包的logger可以不用配置。知道了这个继承特性之后,如果我们要aty.log.dao包下所有的类都打印到同一个日志文件,那么可以进行如下配置。

  1. # a package appender
  2. log4j.logger.aty.log.dao=DEBUG, daoAppender
  3. log4j.appender.daoAppender=org.apache.log4j.FileAppender
  4. log4j.appender.daoAppender.Threshold=warn
  5. log4j.appender.daoAppender.ImmediateFlush=true
  6. log4j.appender.daoAppender.Append=true
  7. log4j.appender.daoAppender.File=c:/dao.log
  8. log4j.appender.daoAppender.layout=org.apache.log4j.PatternLayout
  9. log4j.appender.daoAppender.layout.ConversionPattern=[%d{yyyy-MM-dd HH:mm:ss,SSS}]-[%t]-[%X{ip}]-[%l]-[%p] %m%n
  10. log4j.additivity.aty.log.dao= false

可以看到:利用logger的继承特性,我们可以很容易将不同的日志打印到不同的文件中,这样可以避免各种日志混杂在一起。如果我们想将A.java中有的日志打印到a.log中,有的日志打印到控制台上也是可以的。

  1. package aty.log.service;
  2. import org.apache.log4j.Logger;
  3. public class A {
  4. private static Logger logger = Logger.getLogger(A.class);
  5. private static Logger rootLogger = Logger.getRootLogger();
  6. public void run() {
  7. logger.error("log in A.java");
  8. rootLogger.error("a to console.");
  9. }
  10. }

最后提一下:log4j默认是使用类名或者包名来查找logger的。log4j允许我们自己定义logger的名称。

  1. package aty.log.service;
  2. import org.apache.log4j.Logger;
  3. public class A {
  4. private static Logger logger = Logger.getLogger("atyAlogger");
  5. public void run() {
  6. logger.error("log in A.java");
  7. }
  8. }
  1. #use atyAlogger as appender name.
  2. log4j.logger.atyAlogger=DEBUG, testA
  3. log4j.appender.testA=org.apache.log4j.FileAppender
  4. log4j.appender.testA.Threshold=warn
  5. log4j.appender.testA.ImmediateFlush=true
  6. log4j.appender.testA.Append=true
  7. log4j.appender.testA.File=c:/a.log
  8. log4j.appender.testA.layout=org.apache.log4j.PatternLayout
  9. log4j.appender.testA.layout.ConversionPattern=[%d{yyyy-MM-dd HH:mm:ss,SSS}]-[%t]-[%X{ip}]-[%l]-[%p] %m%n
  10. #avoid print to parent logger.
  11. log4j.additivity.atyAlogger= false

一般来说,我们都是使用默认的名称,这样虽然将类名和包名在log4j.properties写死了。但是方便java类里面获取logger。因为各个类都是根据自己类的全限定名来查找logger的,具体日志到底打印到哪里去,由配置文件决定。

log4j框架logger的继承关系以及使用场景的更多相关文章

  1. Swing框架的继承关系

    ---------------siwuxie095 Java SE 8 (截止 2017/4/1 最新)在线 API 文档: http://docs.oracle.com/javase/8/docs/ ...

  2. ABP VNext框架基础知识介绍(1)--框架基础类继承关系

    在我较早的时候,就开始研究和介绍ABP框架,ABP框架相对一些其他的框架,它整合了很多.net core的新技术和相关应用场景,虽然最早开始ABP框架是基于.net framework,后来也全部转向 ...

  3. 重新想象 Windows 8 Store Apps (16) - 控件基础: 依赖属性, 附加属性, 控件的继承关系, 路由事件和命中测试

    原文:重新想象 Windows 8 Store Apps (16) - 控件基础: 依赖属性, 附加属性, 控件的继承关系, 路由事件和命中测试 [源码下载] 重新想象 Windows 8 Store ...

  4. 如何自建appender扩展Log4j框架

    1.log4j 概述 log4j 环境包括三个主要组件: logger(日志记录器):控制要启用或禁用哪些日志记录语句.可以对日志记录器指定如下级别: ALL . DEBUG . INFO . WAR ...

  5. 使用IntelliJ IDEA查看类的继承关系图形

    最近正好也没什么可忙的,就回过头来鼓捣过去的知识点,到Servlet部分时,以前学习的时候硬是把从上到下的继承关系和接口实现记得乱七八糟. 这次利用了IDEA的diagram,结果一目了然,也是好用到 ...

  6. iOS学习——iOS 整体框架及类继承框架图

    整理自:IOS 整体框架类图值得收藏 一 整体框架 在iOS开发过程中,对iOS的整理框架的了解和学习是必不可少的一个环节,今天我们就好好来了解一下iOS的整体框架.首先贴一个关于iOS的框架介绍:i ...

  7. phpstorm查看类的继承关系

    在看一些框架源码时,有些类有很多的继承或者接口,有一款神奇的帮助很重要 选中一个类文件,右键,选择diagrams->show diagrams 即可得到类的继承关系,如上右图 使用函数 fun ...

  8. java集合继承关系图

    面向对象语言对事物的体现都是以对象的形式,所以为了方便对多个对象的操作,就对对象进行存储,集合就是存储对象最常用的一种方式. 数组虽然也可以存储对象,但长度是固定的:集合长度是可变的,数组中可以存储基 ...

  9. MVC值提供组件ValueProvider的继承关系

    MVC请求过程中中各组件调用顺序:值提供组件(IValueProvider)->模型绑定组件(IModelBinder)->模型验证组件 值提供组件接口 public interface ...

随机推荐

  1. solr的schema.xml配置文件关键词意义

    fieldType:配置扩展的分析器analyzer:具体的分析器的全路径field:配置具体的索引业务字段name:字段的名称type:指定使用哪种分析器域:StringField,textFiel ...

  2. 转发与重定向的区别(forward与redirect的区别)

    转发:服务器接收到客户端的请求后,在服务器内部传递的过程.最后回复结果给客户端. 重定向:服务器接收到客户端的请求后,回复一个新url给客户端,客户端跳转新url.

  3. Silverlight & Blend动画设计系列六:动画技巧(Animation Techniques)之对象与路径转化、波感特效

    当我们在进行Silverlight & Blend进行动画设计的过程中,可能需要设计出很多效果不一的图形图像出来作为动画的基本组成元素.然而在设计过程中可能会出现许多的问题,比如当前绘制了一个 ...

  4. java 的数据库操作--JDBC

    一.java与数据库的交互 1.jdbc:java data base connectivity,java数据库连接.java的JDBC操作主要通过操作两个类进行连接操作:Connection 和 S ...

  5. 解决JVM内存溢出问题

    今天遇到了一个问题,当我在增加配置文件(*.xml)内容的时候,重新启动tomcat6时,控制台报错:java.lang.StackOverflowError: 即,栈溢出错误. 内存溢出,即程序运行 ...

  6. Redis学习笔记1 -- 单机环境时分布式锁的使用

    使用第三方开源组件Jedis实现Redis客户端,且只考虑Redis服务端单机部署的场景. 前言 分布式锁一般有三种实现方式:1. 数据库乐观锁:2. 基于Redis的分布式锁:3. 基于ZooKee ...

  7. python全栈开发之路

    一.Python基础 python简介 python数据类型(数字\字符串\列表) python数据类型(元组\字典) python数据类型(集合) python占位符%s,%d,%r,%f prin ...

  8. Regular Expression学习笔记

    正则写法 var re = /a/;//简写 /.../里不能为空,因为会误以为是注释: var re = new RegExp('a'); 新建一个RegExp对象:和新建Array对象,Objec ...

  9. Python基础-面向过程编程实现Linux下cat -rl ‘dir’ |grep ‘keywords’ 功能

    函数是Python内建支持的一种封装,我们通过把大段代码拆成函数,通过一层一层的函数调用,就可以把复杂任务分解成简单的任务,这种分解可以称之为面向过程的程序设计.函数就是面向过程的程序设计的基本单元. ...

  10. 05_zookeeper的ACL

    [ACL概述] ACL:access control Lists,权限控制. * 针对节点可以设置相关的读写等权限,目的是为了保障数据安全性. * 权限permissions可以指定不同的权限范围以及 ...