百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

Java日志之Slf4j,Log4J,logback原理总结

yuyutoo 2024-10-16 15:41 6 浏览 0 评论

几乎任何应用,一定是需要日志的。

那么,面对种类繁多的日志框架和配置,我们该何去何从?

1.前奏:我是在研究mybatis源码的过程中才意识到需要搞明白日志原理这回事,因为mybatis(和一些其他开源框架,比如rocketmq)都有自己的日志系统,他们在框架内部都使用的是自己的日志API,那么,为什么他们不像我们平常那样配置一个log4j呢?根本原因我也不太清楚,不过我猜测可能有这么一些理由,这些框架比较老,当初还没有slf4j这种事实上的标准,另一方面,有一些特殊的定制化的日志。彻底研究清楚mybatis的日志系统之后,个人觉得这一块设计得不太好,至少今天看来,不太优雅,因为本来一个slf4j就能搞定所有,非得在源码中加入自己的org.apache.ibatis.logging这个包,里面包含一些适配器,虽然代码并不复杂,但是有点多此一举。

2.原理:slf4j是标准,也是门面,他对用户提供统一的API,而下方对接各个日志框架。这有点类似JVM,我们Java开发者使用统一的API,而JVM对接各个操作系统。严格意义上说slf4j自身并不提供日志具体实现。图片来自:https://www.cnblogs.com/hanszhao/p/9754419.html

3.slf4j采用的是SPI机制,指定一个标准的目录结构:org.slf4j.impl.StaticLoggerBinder,然第三方的框架都必须存在一个这样的类,用于和slf4j建立关系,比如slf4j-simple.jar,logback,这两个直接实现了slf4j的接口,而对于log4j这种则需要一个中间适配器slf4j-log4j12。于是乎,当调用slf4j的Logger logger = LoggerFactory.getLogger(XXX.class)的时候,虽然使用的是slf4j的api,但是真正输出日志的是具体的日志框架,这样子做的好处就是,当某一天你希望更换日志框架了,只需要把具体日志框架的jar包替换掉,不需要更改任何一行代码,就能实现日志框架的切换。

4.slf4j是如何发现具体日志框架的,这就得意于spi机制,前面说每个日志框架都需要存在一个org.slf4j.impl.StaticLoggerBinder类,log4j则是通过中间适配器slf4j-log4j12。当调用LoggerFactory.getLogger的时候,就会去classpath中寻找StaticLoggerBinder这个类,如果不存在或者存在超过1个,那么会报错,classpath有且只能存在一个StaticLoggerBinder类。

5.分析mybatis的日志框架:mybatis有一套属于自己的日志系统,日志api是:Log log = LogFactory.getLog(xxx.class),于此同时,封装了几个主流的日志框架适配器,包括:SLF4J | LOG4J | LOG4J2 | JDK_LOGGING | COMMONS_LOGGING | STDOUT_LOGGING | NO_LOGGING,当调用Log log = LogFactory.getLog(xxx.class)时,会初始化众多适配器中的一个,可以在mybatis的配置文件中通过logImpl指定具体的一个,如果不指定那么默认使用SLF4J,因为这里在LogFactory类中的静态代码快第一个就是SLF4J:

 static {
 tryImplementation(LogFactory::useSlf4jLogging);
 tryImplementation(LogFactory::useCommonsLogging);
 tryImplementation(LogFactory::useLog4J2Logging);
 tryImplementation(LogFactory::useLog4JLogging);
 tryImplementation(LogFactory::useJdkLogging);
 tryImplementation(LogFactory::useNoLogging);
 }

假设使用默认配置,那么就会初始化Slf4jImpl类,这个类内部有个代理log,这个代理log就是Logger logger = LoggerFactory.getLogger(clazz),这就回归到slf4j的标准使用方式上面来了,mybatis打印日志,其实就是代理对象在打印,而代理对象就是classpath中配置的具体日志框架。

6.分析log4j是如何与slf4j整合的:前面说到,要使用log4j就必须引入slf4j-log4j12这个jar包,而这个jar包中同样存在一个StaticLoggerBinder类,当我们调用LoggerFactory.getLogger(clazz)的时候,同样是初始化StaticLoggerBinder,然后调用利用ILoggerFactory创建一个log4j的Logger实例,代码如下:

 1 public class Log4jLoggerFactory implements ILoggerFactory {
 2 
 3 // key: name (String), value: a Log4jLoggerAdapter;
 4 ConcurrentMap<String, Logger> loggerMap;
 5 
 6 
 7 public Log4jLoggerFactory() {
 8 loggerMap = new ConcurrentHashMap<String, Logger>();
 9 }
10 
11 /*
12 * (non-Javadoc)
13 * 
14 * @see org.slf4j.ILoggerFactory#getLogger(java.lang.String)
15 */
16 public Logger getLogger(String name) {
17 Logger slf4jLogger = loggerMap.get(name);
18 if (slf4jLogger != null) {
19 return slf4jLogger;
20 } else {
21 org.apache.log4j.Logger log4jLogger;
22 if(name.equalsIgnoreCase(Logger.ROOT_LOGGER_NAME))
23 log4jLogger = LogManager.getRootLogger();
24 else
25 log4jLogger = LogManager.getLogger(name);
26 
27 Logger newInstance = new Log4jLoggerAdapter(log4jLogger);
28 Logger oldInstance = loggerMap.putIfAbsent(name, newInstance);
29 return oldInstance == null ? newInstance : oldInstance;
30 }
31 }
32 }

最关键的一行就是第27行Logger newInstance = new Log4jLoggerAdapter(log4jLogger),slf4j的Logger对象实际上是一个log4j的适配器对象(也是代理对象),当slf4j调用比如debug方法的时候,实际上是代理对象(也就是真实的log4j对象)在调用debug方法。

相关推荐

从零搭建高可用的 MySQL 主从复制架构(基于 Linux 实战指南)

背景在生产环境中,单点MySQL数据库容易成为性能瓶颈或单点故障源。搭建MySQL主从复制架构,可以实现读写分离、高可用,提升系统的整体稳定性与扩展性。...

国外大神成功让Nexus4吃上安卓6.0:基本可正常使用

IT之家讯10月9日消息谷歌已经于10月6日正式开启了Nexus设备Android6.0Marshmallow系统的OTA升级推送。根据之前报道的消息,老一批的Nexus手机如Nexus4/Ne...

急死!CPU被挖矿了,却找不到哪个进程

CPU起飞了最近有朋友在群里反馈,自己服务器的CPU一直处于高占用状态,但用...

甜甜的安卓5.0却让手机ROOT难度大大增加

IT之家(www.ithome.com):甜甜的安卓5.0却让手机ROOT难度大大增加对设备进行ROOT,毫无疑问,这是安卓最美丽的地方之一,不管是对于消费者来说还是开发者。Root意味着掌握更多的权...

Linux基础知识(linux基础知识点及答案)

系统目录结构/bin:命令和应用程序。/boot:这里存放的是启动Linux时使用的一些核心文件,包括一些连接文件以及镜像文件。/dev:dev是Device(设备)的缩写,该目录...

Linux 内核 6.15 发布:内存、网络、文件系统全面升级!

核心增强:性能与安全双飞升!Linux内核6.15的正式版!虽然因一个临门一脚的Bug晚了几小时,但最终还是带着一堆硬核更新闪亮登场!...

AlmaLinux 9.6 发布,新增功能亮点纷呈!

距离上一版本AlmaLinux9.5发布六个月后,基于5.14内核的AlmaLinux正式宣布其企业级Linux发行版的9.x系列第六个更新——AlmaLinux9.6(Sag...

理解Linux下的SELinux(linux seccomp)

理解Linux下的SELinux长久以来,每当遇到授权问题或者新安装的主机,我的第一反应是通过setenforce0命令禁用SELinux,来减少产生的权限问题,但是这并不是一个良好的习惯。这篇文章...

3个简单实用的网址导航网站(简洁的网站导航)

在我们使用电脑上网的时候经常会访问某些常用的网站,每一次都去通过搜索访问就比较浪费时间,添加在浏览器收藏夹不方便在其他电脑使用。找一个好用的网址导航网站就可以帮我们把所有常用的网址集合在一个页面,方便...

整点不一样的网站制作教程,教你怎么用网站模板制作网站#...

网站制作教程整点不一样的网站。不要再问我网站制作教程了,今天给你整个怎么用网站模板制作网站的教程。·1、登录账号进入后台。·2、选择模板。自助建站平台通常提供各种各样的网站模板,可以根据自己的需求和喜...

5个最好的外贸独立站模板,让你的网站更加专业

作为外贸行业从业者,一个专业且具有吸引力的网站是必不可少的。然而,建立一个专业的网站需要耗费大量的时间和精力,尤其是在设计和开发方面。为了帮助您缩短网站建设的时间和成本,以下是5个最好的外贸独立站模板...

网站建设模板 **网站建设模板:全面指南与创意构思*

网站建设模板**网站建设模板:全面指南与创意构思**随着互联网技术的迅猛发展,网站已成为企业、机构和个人展示自身形象、传递信息、实现交流的重要平台。本文将详细介绍网站建设的基本模板,并提供创意...

原地封神!一个只用套模板即可制作电子相册的网站

对于忙碌的年轻人来说,一键操作的模板意味着无需复杂的操作步骤,就能轻松制作出精美的电子相册。但是一个好的工具也是事关重要,最近发现了一款非常适合年轻人的模板---FLBOOK在线制作电子杂志平台,只需...

跨屏建站网kpfree免费网站模板2023.1.14发布更新

跨屏建站网kpfree免费网站模板2023.1.14发布更新,摒弃了之前的卡片式设计,采用了移动优先的设计原则,简化了页面设计风格,优化了代码,优化了图片质量,确保网页打开速度。砍掉了一些花哨而无用的...

响应式大型电子企业集团类网站模板源码-青柠资源网qnziyw.cn

模板信息:模板编号:10964模板编码:UTF8模板颜色:红色模板分类:科技、电子、数码设备适合行业:电子设备类企业模板介绍:本模板自带eyoucms内核,无需再下载eyou系统,原创设计、手工书写D...

取消回复欢迎 发表评论: