Spring源码学习:第2步--使用SLF4j+Log4j日志框架替换掉其自身的commons-logging日志框架
正如Spring官方文档所述,其底层的实现选择了commons-logging作为日志框架。这一“失足”性的选择,竟连Spring自身都抱怨。但是,谁叫Spring如此优秀呢,即使有一点小瑕疵,人家也得完美的兼容下去。所以,直到目前的版本,commons-logging仍然是默认的日志框架。
但是,大多数的应用开发(也包括我自己做过的应用),都没有选择commons-logging这个日志框架,而是SLF4j 或者 Log4j 之类的。
那有没有办法,像其它很多开源框架一样,替换掉其默认的commons-logging呢?显然,是有的!
从Spring官方文档中可以看出,它其实也只有 spring-core 这个模块依赖了commons-logging,所以,替换掉它就可以了。
以使用SLF4j框架来替换为例,具体的替换步骤如下(参照官网):
- 从 spring-core 的依赖中排除掉 commons-logging 依赖。
- 添加 jcl-over-slf4j 依赖。
- 添加 slf4j 以及 log4j 依赖。
详细说明如以下截图所示:
第一步:排除对 commons-logging 的依赖。
第二步:添加接管 commons-logging 的依赖。
刚开始的时候,自己也不理解为什么添加这样一个特殊的依赖且不用修改源码就可以让Spring的日志自动转到 slf4j 上来呢?其实,不难理解,这真的是一个“接管”包,因为其内部的Log接口以及最常用的LogFactory类都与原来 commons-logging 中的一样!所以,底层代码就完全不需要改变,同时,输出日志信息的方法调用就自动转到了这个 jcl-over-slf4j 包的方法上。这样就实现了“接管”。
第三步:添加 slf4j 及 log4j 依赖。
通过这样三步,commons-logging 的日志方法调用就全部转嫁到 jcl-over-slf4j,再到 slf4j 最后到 log4j 上面了。
我没有直接使用 Log4J 来接管,因为 slf4j 比 log4j 更好一些(主要体现在占位符以及Debug和Trace日志不用全部拼接字符串的逻辑上)。
现在Spring官方文档的格式好像改版了,没有以前那样可以直接在一个Html页面中显示出来(进而可以保存为PDF并打印)。所以,在此贡献一个我当年从单个Html页面中保存下来的PDF文档吧。同时,上述接管 commons-logging 的原文也可以在第2章中找到。
本PDF是Spring的 4.3.5.RELEASE 版本。
============================================================================================================================
上面的接管其实还比较简单,而且也能看到,这是在Spring源码之外且基于Maven来管理的。问题是:如何在Spring源码中替换呢?
Spring源码使用了Gradle来管理,所以,应该能够想到,是通过直接修改 build.gradle 文件来完成的。
我自己在GitHub上Checkout出来后使用的是 4.2.5.RELEASE 版本,可以看到,这个版本的 build.gradle 文件有1400+行。自己没有系统学习和使用过Gradle,于是,依照着对maven的理解,开始改吧。
依旧可以清晰地看到,Spring下的十几个模块在 build.gradle 文件中都有类似于 project("spring-core") 这样的关键字。在这其中还可以看到 dependencies 关键字,不用想,这和Maven中的 dependencies 一样,就是依赖了。
明显,commons-logging 的依赖位列其中。所以,想办法将它替换成 slf4j 应该就可以了。
可是,咋个改呢??
开始的尝试(第一次)是:
- 删除 commons-logging 这一行的依赖
- 添加 jcl-over-slf4j 依赖、slf4j-api 依赖以及 slf4j-log4j12 依赖(都是 compile 范围)
- 将 optional 的 log4j 依赖改为 compile
为什么要这样做呢?很简单,完全是从上面的Maven中照搬过来的!然而,悲剧发生了,改了后它就是不正常。更奇怪的是,它会报 spring-context 对 spring-beans-groovy 的依赖找不到!
完全无解,憋了好久也没想出个办法。。。
完全无解,憋了好久也没想出个办法。。。
完全无解,憋了好久也没想出个办法。。。
没办法,只有尝试了。整个操作的现象就是,添加了上述依赖后,它就报 spring-context 对 spring-beans-groovy 的依赖找不到!于是,我少少地添加呗,再不济,一个一个地添加呗。
现在的尝试(第N次)是:
- 删除 commons-logging 这一行的依赖——因为要排除掉 commons-logging 的依赖,所以这一步是必须的!
- 添加 jcl-over-slf4j 依赖、slf4j-api 依赖以及 slf4j-log4j12 依赖(都是 compile 范围)——因为排除 commons-logging 后,源码中直接报错,没有需要的类,所以,这些依赖必须加进来。
只操作了上述两步,再一试,还是报错,但错误不一样了,而且一看就懂,大概就是说找不到类,而且是 log4j 中的 Log 类(其实是接口),而且错误是在 spring-context 是报出来的,于是,尝试再在 spring-context 的依赖中添加 log4j 的依赖。结果,成了!
对Gradle不熟悉,所以,虽然问题是解决了也可以正常运行了,但是,RootCause还是不清楚。而且从上述截图中也可以看到 spring-core 的依赖中已经有 optional 的 log4j 依赖了,而我之前是直接把这个 optional 改成 compile 了,会不会有影响?而最后能正常运行的时候,这个 optional 的 log4j 仍然是存在的,只是在 spring-context 中添加了 compile 的 log4j。
献上自己从GitHub Fork并“践踏”过的Spring源码地址:https://github.com/cyhbyw/spring-framework/tree/Branch_v4.2.5.RELEASE
Spring源码学习:第2步--使用SLF4j+Log4j日志框架替换掉其自身的commons-logging日志框架的更多相关文章
- spring源码学习之路---深入AOP(终)
作者:zuoxiaolong8810(左潇龙),转载请注明出处,特别说明:本博文来自博主原博客,为保证新博客中博文的完整性,特复制到此留存,如需转载请注明新博客地址即可. 上一章和各位一起看了一下sp ...
- spring源码学习之路---IOC初探(二)
作者:zuoxiaolong8810(左潇龙),转载请注明出处,特别说明:本博文来自博主原博客,为保证新博客中博文的完整性,特复制到此留存,如需转载请注明新博客地址即可. 上一章当中我没有提及具体的搭 ...
- Spring源码学习-容器BeanFactory(四) BeanDefinition的创建-自定义标签的解析.md
写在前面 上文Spring源码学习-容器BeanFactory(三) BeanDefinition的创建-解析Spring的默认标签对Spring默认标签的解析做了详解,在xml元素的解析中,Spri ...
- Spring源码学习-容器BeanFactory(三) BeanDefinition的创建-解析Spring的默认标签
写在前面 上文Spring源码学习-容器BeanFactory(二) BeanDefinition的创建-解析前BeanDefinition的前置操作中Spring对XML解析后创建了对应的Docum ...
- Spring源码学习-容器BeanFactory(一) BeanDefinition的创建-解析资源文件
写在前面 从大四实习至今已一年有余,作为一个程序员,一直没有用心去记录自己工作中遇到的问题,甚是惭愧,打算从今日起开始养成写博客的习惯.作为一名java开发人员,Spring是永远绕不过的话题,它的设 ...
- Spring源码学习笔记9——构造器注入及其循环依赖
Spring源码学习笔记9--构造器注入及其循环依赖 一丶前言 前面我们分析了spring基于字段的和基于set方法注入的原理,但是没有分析第二常用的注入方式(构造器注入)(第一常用字段注入),并且在 ...
- Spring 源码学习笔记11——Spring事务
Spring 源码学习笔记11--Spring事务 Spring事务是基于Spring Aop的扩展 AOP的知识参见<Spring 源码学习笔记10--Spring AOP> 图片参考了 ...
- Spring源码学习笔记12——总结篇,IOC,Bean的生命周期,三大扩展点
Spring源码学习笔记12--总结篇,IOC,Bean的生命周期,三大扩展点 参考了Spring 官网文档 https://docs.spring.io/spring-framework/docs/ ...
- Spring源码学习
Spring源码学习--ClassPathXmlApplicationContext(一) spring源码学习--FileSystemXmlApplicationContext(二) spring源 ...
- Spring源码学习-容器BeanFactory(二) BeanDefinition的创建-解析前BeanDefinition的前置操作
写在前面 上文 Spring源码学习-容器BeanFactory(一) BeanDefinition的创建-解析资源文件主要讲Spring容器创建时通过XmlBeanDefinitionReader读 ...
随机推荐
- Java语言的特性
一.跨平台 借助虚拟机,程序不经修改即可在不同硬件或者软件平台上运行.源代码级(C,C++源码会重新编译),目标代码级(Java). 二.面向对象 以对象为基本单位,使得程序开发变得简单易用,拓展更方 ...
- UVA - 11995 I Can Guess the Data Structure!(模拟)
思路:分别定义栈,队列,优先队列(数值大的优先级越高).每次放入的时候,就往分别向三个数据结构中加入这个数:每次取出的时候就检查这个数是否与三个数据结构的第一个数(栈顶,队首),不相等就排除这个数据结 ...
- 计蒜客的一道题dfs
这是我无聊时在计蒜客发现的一道题. 题意: 蒜头君有一天闲来无事和小萌一起玩游戏,游戏的内容是这样的:他们不知道从哪里找到了N根不同长度的木棍, 看谁能猜出这些木棍一共能拼出多少个不同的不等边三角形. ...
- Python多进程----从入门到放弃
Python多进程 (所有只写如何起多进程跑数据,多进程数据汇总处理不提的都是耍流氓,恩,就这么任性) (1)进程间数据问题,因为多进程是完全copy出的子进程,具有独立的单元,数据存储就是问题了 ( ...
- 从 PHP 到 Java
* { color: #3e3e3e } body { font-family: "Helvetica Neue", Helvetica, "Hiragino Sans ...
- java调用shell脚本,并获得结果集的例子
/** * 运行shell脚本 * @param shell 需要运行的shell脚本 */ public static void execShell(String shell){ try { Run ...
- JS分号 惹的坑
JS中会自动清除句子和句子之间的空格以及tab缩进, 这样就可以允许用户编写的代码更加随性和更加可读, 在该行代码解析的时候如果该行代码可以解析, 就会在该行代码最后自动填写分号,如果该行代码无法解析 ...
- 慢慢来写SpringMVC基本项目
首先新建一个maven项目. 这里选用webapp的.在点击next弹出的界面的groupID和artifactID自己定义憋.好了,这个第一步完成.创建完可能会有个红叉在项目上, 这个只需要在pom ...
- 图像处理------基于Otsu阈值二值化
一:基本原理 该方法是图像二值化处理常见方法之一,在Matlab与OpenCV中均有实现. Otsu Threshing方法是一种基于寻找合适阈值实现二值化的方法,其最重 要的部分是寻找图像二值化阈值 ...
- 命令行工具osql.exe使用
目标: 快速在21个库修改Test表的某条记录,这几个库都分别在不同的服务器上. 通常会想到,到每个库都执行一下语句不就好了吗?这个数据库切换来切换去,挺麻烦了,通过命令行工具osql.exe就可以快 ...