java的日志知识
java常用的日志有以下几种 :
一、jdk自带的java.util.logging包下的日志功能, 不常用。
二、commons-logging + log4j 的搭配 。log4j是日志功能的具体实现,而commons-logging则是日志接口的声明,它的出现也是为了解决应用和具体的日志框架解耦合的问题,它采用的是运行时动态绑定的方式来决定使用哪个日志框架。 什么是动态绑定 ?参考commons-logging-1.1.3的org.apache.commons.logging.LogFactory类的方法:
public static Log getLog(Class clazz) throws LogConfigurationException {
return getFactory().getInstance(clazz);
}
关键在getFactory()返回的LogFactory(日志工厂) 是什么 !
我们进一步看获取日志工厂的方法:
public static LogFactory getFactory() throws LogConfigurationException
从419-646行:代码比较长不列出,阐述如下:
1、获取线程上下文ClassLoader(默认是加载应用的ClassLoader)。
2、看线程上下文ClassLoader是否有缓存的LogFactory,有就直接返回LogFactory 。
3、找classpath下面的commons-logging.properties ,如果use_tccl属性为false,则不使用Thread ContextClassLoader .默认use_tccl 为true ;.
4、找是否有org.apache.commons.logging.LogFactory 这个系统配置项,利用Thread ContextClassLoader 加载org.apache.commons.logging.LogFactory ,并创建一个LogFactory实例(在后面要描述的jcl-over-slf4j包和commons-logging包里面都有这个类。)
5、如果还找不到,则找包含了META-INF/services/org.apache.commons.logging.LogFactory 这个文件的Jar包,找到了就用这个文件里面的LogFactory类名创建LogFactory实例
6、如果还找不到,则找commons-logging.properties 文件,利用属性文件里面的org.apache.commons.logging.LogFactory 属性获取LogFactory类并创建LogFactory实例。
7、还找不到,则找org.apache.commons.logging.impl.LogFactoryImpl 这个类 创建实例。
8、创建好LogFactory实例以后,会
cacheFactory(contextClassLoader, factory);
把LogFactory对象和线程上下文ClassLoader在map中关联起来,加速LogFactory对象的获取。
从以上的分析来看,我们假设一种简单的场景,
没有org.apache.commons.logging.LogFactory 这个系统配置项,classpath下没有包含META-INF/services/org.apache.commons.logging.LogFactory 这个文件的Jar包、没有commons-logging.properties 文件,只有commons-logging和Log4j的jar .
此时会使用org.apache.commons.logging.impl.LogFactoryImpl 这个类(在commons-logging的jar 包里面) 来创建Log实例,而LogFactoryImpl在获取Log类的时候,会参照下面一个顺序:
private static final String[] classesToDiscover = {
LOGGING_IMPL_LOG4J_LOGGER,
"org.apache.commons.logging.impl.Jdk14Logger",
"org.apache.commons.logging.impl.Jdk13LumberjackLogger",
"org.apache.commons.logging.impl.SimpleLog"
};
来使用某一个日志器, 可以看到默认就是使用log4j日志的。
至此,我们弄清了commons-logging的动态绑定机制。
但是这种机制的问题在哪儿呢,由于它使用了ClassLoader寻找和载入底层的日志库, 导致了象OSGI这样的框架无法正常工作,因为OSGI的不同的插件使用自己的ClassLoader。 OSGI的这种机制保证了插件互相独立,然而却使Apache Common-Logging无法工作。
三、slf4j + logback的搭配.
slf4j和logback是同一个人开发的,所以不像slf4j + log4j搭配使用时,还需要加上一个所谓的适配器jar包(比如:slf4j-log4j.jar,适配器包的作用就是通过class composition的适配方式把log4j的日志转换成slf4j的接口)。
和commons-logging在运行时确定日志框架不同,slf4j采用的方式是在编译时静态绑定真正的log库, 它的原理是:
(1)用ClassLoader找包含了org.slf4j.impl.StaticLoggerBinder类的Jar包,这个Jar就包含在logback-classic.jar、slf4j-log4j等包里面 ,因此,如果在classpath下面
同时包含了这些jar的话, 程序将会出现一个警告,提示classpath下面有多个slf4j的绑定 。这也是我们在使用slf4j日志时需要注意的问题,不过slf4j并不会报错, 而是选择一个
jar中的StaticLoggerBinder,所以在使用的时候要特别注意不能同时包含logback-classic 、 slf4j-log4j、slf4j-jdk等桥接包 。
我们假设使用的是logback做日志框架 ,这时会拿到logback-classic.jar里面的org.slf4j.impl.StaticLoggerBinder ,这个StaticLoggerBinder获取到日志工厂以后,
利用日志工厂获取到ch.qos.logback.classic.Logger, 接下来使用的就是logback的日志了。
而如果是classpath下面包含的是slf4j-log4j这个桥接包, 那么拿到的就是Log4j的LogFactory,从而也就用到了log4j的日志。
四、既然slf4j的静态绑定方式解决了commons-logging动态绑定方式在运行时可能拿不到日志接口实现类的问题,而且号称效率比log4j要更好(为什么更好,后续还会深入分析) 那直接都换成slf4j+logback的日志方式不就行了么,但现实是很多的应用之前都是建立在commons-logging+log4j的日志方式上的,有什么办法不改动应用的代码,达到commons-logging日志转到slf4j的目的么?把commons-logging.jar替换掉就好了,看下图:
具体的原理是什么呢? 以LogFactory.getLog("loggerName")为例:
1、org.apache.commons.logging.LogFactory类被jcl-over-slf4j包里面的同包同名类替换掉了。
2、获取到的日志工厂是一个SLF4jLogFactory ,这个日志工厂在获取org.apache.commons.logging.Log 实例的时候,先基于前面描述的slf4j静态绑定机制,拿到了一个org.slf4j.Logger,然后用一个适配器类做接口转换,把slf4j的日志转换成commons-logging的日志器。
总结下来,我们有以下几点启发:
1、适配器模式可以帮助我们做各种接口包的无缝集成。
2、复杂的java应用还是在classloader机制上做文章,即使是slf4j的静态绑定机制 ,其实也是在编译时检查了classpath下是否有多个jar包包含StaticLoggerBinder类,
真正运行的时候由线程上下文的classloader(默认是app classloader)来加载这个StaticLoggerBinder类而已。
3、要分清Java日志系统里面各种包的作用:
1)日志接口包:commons-logging , slf4j-api
2) 日志框架包:log4j, logback,
3) 日志适配器包:slf4j-log4j,slf4j-jdk
4) 把某一个日志接口转换到另一个日志接口的桥接包:jcl-over-slf4j,log4j-over-slf4j 等。
java的日志知识的更多相关文章
- 【Java】日志知识总结和经常使用组合配置(commons-logging,log4j,slf4j,logback)
Log4j Apache的一个开放源码项目,通过使用Log4j,我们能够控制日志信息输送的目的地是控制台.文件.GUI组件.甚至是套接口服务 器.NT的事件记录器.UNIX Syslog守护进程等.用 ...
- JAVA主流日志梳理
JAVA主流日志梳理 引入 历史故事 Log4j - JDK1.3及以前 JUL - JDK1.4 JCL - 日志门面commons-logging的出现 SLF4j - 可能是最好的日志框架 lo ...
- 探索Java的日志世界
探索Java的日志世界 本文的思维导图 一.主题 打开日志的大门,探索的Java日志世界 二.目标 了解常用的日志框架 掌握日志框架的选择和使用以及开发规范 了解日志框架中的一些设计思想 三.内容 1 ...
- 细说Java主流日志工具库
概述 在项目开发中,为了跟踪代码的运行情况,常常要使用日志来记录信息. 在Java世界,有很多的日志工具库来实现日志功能,避免了我们重复造轮子. 我们先来逐一了解一下主流日志工具. java.util ...
- Java主流日志工具库
在项目开发中,为了跟踪代码的运行情况,常常要使用日志来记录信息.在Java世界,有很多的日志工具库来实现日志功能,避免了我们重复造轮子.我们先来逐一了解一下主流日志工具. 1.java.util.lo ...
- Java 多线程——基础知识
java 多线程 目录: Java 多线程——基础知识 Java 多线程 —— synchronized关键字 java 多线程——一个定时调度的例子 java 多线程——quartz 定时调度的例子 ...
- Java 标准日志工具 Log4j 的使用(附源代码)
源代码下载 Log4j 是事实上的 Java 标准日志工具.会不会用 Log4j 在一定程度上可以说是衡量一个开发人员是否是一位合格的 Java 程序员的标准.如果你是一名 Java 程序员,如果你还 ...
- Java面试必备知识
JAVA面试必备知识 第一,谈谈final, finally, finalize的区别. 第二,Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可 ...
- Java自定义日志输出文件
Java自定义日志输出文件 日志的打印,在程序中是必不可少的,如果需要将不同的日志打印到不同的地方,则需要定义不同的Appender,然后定义每一个Appender的日志级别.打印形式和日志的输出路径 ...
随机推荐
- cogs1772 [国家集训队2010]小Z的袜子
沉迷于文化的我N年没更blog了...(\(N \in (0,1)\)) 然后回到机房就沉迷于 \(generals.io\) 无法自拔...QAQ 然后想打一遍splay(然后是LCT),然后放弃了 ...
- target属性用于返回最初触发事件的DOM元素。
target属性用于返回最初触发事件的DOM元素. 在HTML文档中,我们为<p>元素绑定点击事件("click"),由于DOM元素的事件冒泡机制,我们点击<p& ...
- C# 其他的Url 文件的路径转化为二进制流
//将虚拟路径转化为文件的路径然后最后转化为文件流 public ActionResult SaveImage(string path) { var url =System.Web.HttpConte ...
- linux for 循环的小应用
[root@localhost ~]# mkdir -pv /home/data{1..5} # 创建多个目录 以下两种方法类似. for i in {1..5};do echo "&l ...
- React——条件渲染
在React中,你可以创建各种不同的组件,然后根据应用的状态渲染出它们其中的一般部分. 一.用变量存储元素 可以将元素保存到一个变量中,通过为变量赋不同的值去渲染不同的元素 function Logi ...
- JMeter的__threadGroupName使用注意事项
JMeter从4.1版本开始引入了一个新函数"${__threadGroupName}",这个函数的作用是返回当前线程组的名字.${__threadGroupName}的用途也较为 ...
- Android #Android开发环境搭建
Android #Android开发环境搭建 1.下载:Google在国服的官网 https://developer.android.google.cn/index.html 1.点击首页 “ 获取 ...
- nginx 定义的一些状态码
ngx_string(ngx_http_error_494_page), /* 494, request header too large */ ngx_string(ngx_http_erro ...
- SICP读书笔记 2.3
SICP CONCLUSION 让我们举起杯,祝福那些将他们的思想镶嵌在重重括号之间的Lisp程序员 ! 祝我能够突破层层代码,找到住在里计算机的神灵! 目录 1. 构造过程抽象 2. 构造数据抽象 ...
- DevOps on AWS之Cloudformation实践篇
cloudformation入门实践 AWS cloudformation通过模板对AWS云资源进行编排和调用.并且可以通过模板代码层面的修改就可以对现有环境进行升级改造,云端业务的灵活便捷特点展现无 ...