热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

面试官突然的关心日志篇SLF4J

当面试官问到,你们用的什么日志框架,log4j和logback吗?和slf4j什么区别?吓尿了,日志系统也关心

当面试官问到,你们用的什么日志框架,log4j 和logback吗?和slf4j什么区别?吓尿了,日志系统也关心,学起来!



前言

日志重要吗?程序猿排查问题,数据统计分析、故障定位都依赖日志,甚至有的公司直接用日志的输出,经过统计做业务系统的输入。所以日志实在太重要了,大家几乎每天都在用日志,那都知道用的是什么日志框架,以及实现的原理吗?


开胃菜

说Log4j 和 Logback 之前,不得不提SLF4J。

简单说,SLF4J相当于定义了接口,Log4j 和 Logback是具体实现。

SLF4J就是典型的门面模式,什么是门面模式,简单说系统访问入口SLF4J只提供一个门面,具体实现对调用方不可见,SLF4J门面模式画了个简图,如下:

图片

阿里Java开发手册关于日志的第一条


【强制】应用中不可直接使用日志系统(Log4j、Logback)中的API,而应依赖使用日志框架 SLF4J中的API,使用门面模式的日志框架,有利于维护和各个类的日志处理方式统一。



前菜-泡菜豆腐汤(工程实践)

要打印日志,就会引入上面提到的日志框架,就拿我们平时工程中常用的maven来讲,会同时引入Log4j、sfl4j-jdk / Logback等作为日志记录系统,那SLF4J门面是怎么绑定具体的日志系统的呢?下面重点都做了加粗。

第一步
我们先写一个简单例子:新建一个maven工程,在pom中只加入SLF4J的依赖包,


org.slf4jslf4j-api1.7.24

写段打印日志的测试代码, 如下:

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class Slf4jTest {static final Logger logger = LoggerFactory.getLogger(Slf4jTest.class);public static void main(String[] args) {logger.info("验证slf4j");}
}

运行结果如下:


SLF4J: Failed to load class "org.slf4j.impl.StaticLoggerBinder".
SLF4J: Defaulting to no-operation (NOP) logger implementation
SLF4J: See http://www.slf4j.org/codes.html#StaticLoggerBinder for further details.


因为我们现在只引入了门面SLF4J, 没有具体实现, 所有会报找不到StaticLoggerBinder的异常日志。

这个是运行期的报错,编译没问题,因为这个门面的具体实现是运行期绑定的

第二步
我们在pom文件中添加logback的依赖,如下所示

org.slf4jslf4j-api1.7.24ch.qos.logbacklogback-classic1.1.11

运行结果如下:

图片

我们说过SLF4J 是运行期决定绑定哪个具体实现的

现在我们来做个实现,添加多个SLF4J的日志实现,看实现:

org.slf4jslf4j-api1.7.24log4jlog4j1.2.17ch.qos.logbacklogback-classic1.1.11org.slf4jslf4j-simple1.7.25org.slf4jslf4j-log4j121.7.21

运行结果:

图片

SLF4J会提示Classpath 包含多个SLF4J的实现,已经最终从这些中选择的是logback实现。

那问题来了:LoggerFactory如何绑定具体的日志系统的呢?


正餐-韩式烤肉(源码分析)

先说结论

所有实现SLF4J标准的日志系统都需要提供StaticLoggerBinder类,如下图所示:

图片
我们在Class 文件头写下面这段代码时,SLF4J是如何绑定具体日志系统的呢?

private Logger logger = LoggerFactory.getLogger(***.class);

看下LoggerFactory 的代码, 我们可以用IDEA 提供的快捷键Ctrl(Mac是Command)+ Shift +鼠标点击getLogger方法,进实现看。

public static Logger getLogger(Class clazz) {Logger logger = getLogger(clazz.getName());//其他代码return logger;}public static Logger getLogger(String name) {//获取系统LoggerFactoryILoggerFactory iLoggerFactory = getILoggerFactory();return iLoggerFactory.getLogger(name);}

LoggerFactory的getILoggerFactory实现如下图:

这段逻辑就是判断日志系统是否初始化,如果还没初始化,先进行初始化,performInitialization() 只会执行一次。

初始化过程中,INITIALIZATION_STATE是 初始化过程中状态。下面是定义的状态和变量

static final int UNINITIALIZED = 0;static final int ONGOING_INITIALIZATION = 1;static final int FAILED_INITIALIZATION = 2;static final int SUCCESSFUL_INITIALIZATION = 3;static final int NOP_FALLBACK_INITIALIZATION = 4;//日志系统初始化状态static volatile int INITIALIZATION_STATE = UNINITIALIZED;

可以看到INITIALIZATION_STATE是volatile 修饰的,保证多线程初始化日志框架的时候状态的可见性,不会出现多次初始化

我们再看下 performInitialization() 函数,重点就在bind函数, 这里实际上就是绑定具体的日志框架,如下:

private final static void performInitialization() {bind();//就在这个函数实现具体日志系统绑定if (INITIALIZATION_STATE == SUCCESSFUL_INITIALIZATION) {versionSanityCheck();}}

private final static void bind() {try {Set staticLoggerBinderPathSet = null;// skip check under android, see also// http://jira.qos.ch/browse/SLF4J-328//第一步: 如果是Android 则跳过, 否则查找类路径下所有的StaticLoggerBinder实现类if (!isAndroid()) {//划重点,看后面这个函数staticLoggerBinderPathSet = findPossibleStaticLoggerBinderPathSet();reportMultipleBindingAmbiguity(staticLoggerBinderPathSet);}// the next line does the binding//第二步: 可能会有多个StaticLoggerBinder实现类,随机绑定其中一个StaticLoggerBinder.getSingleton();INITIALIZATION_STATE = SUCCESSFUL_INITIALIZATION;//打印实际绑定的reportActualBinding(staticLoggerBinderPathSet);fixSubstituteLoggers();replayEvents();// release all resources in SUBST_FACTORYSUBST_FACTORY.clear();} catch (NoClassDefFoundError ncde) {// ...} catch (java.lang.NoSuchMethodError nsme) {// ...} catch (Exception e) {// ...}}

重点函数 findPossibleStaticLoggerBinderPathSet():

static Set findPossibleStaticLoggerBinderPathSet() {// use Set instead of list in order to deal with bug #138// LinkedHashSet appropriate here because it preserves insertion order// during iterationSet staticLoggerBinderPathSet = new LinkedHashSet();try {ClassLoader loggerFactoryClassLoader = LoggerFactory.class.getClassLoader();Enumeration paths;// 路径:STATIC_LOGGER_BINDER_PATH = org/slf4j/impl/StaticLoggerBinder.class//就是查找classpath下所有 StaticLoggerBinder,限制一定是org.slf4j.impl这个包路径if (loggerFactoryClassLoader == null) {paths = ClassLoader.getSystemResources(STATIC_LOGGER_BINDER_PATH);} else {paths = loggerFactoryClassLoader.getResources(STATIC_LOGGER_BINDER_PATH);}while (paths.hasMoreElements()) {URL path = paths.nextElement();staticLoggerBinderPathSet.add(path);}} catch (IOException ioe) {Util.report("Error getting resources from path", ioe);}return staticLoggerBinderPathSet;}

如果系统中有多个SLF4J实现时,ClassLoader.getResources() 方法会从ClassPath查找到多个StaticLoggerBinder的实现类,通过类加载器加载所有StaticLoggerBinder类(指定包路径,这个也是个知识点,包路径+类名唯一标识一个类, 类加载机制之后文章会讲),

org/slf4j/impl/StaticLoggerBinder.class

如下图所示:

图片

我们看下是不是每个日志框架都有自己的 org/slf4j/impl/StaticLoggerBinder.class


甜点-延伸

如果我们自己要实现一套日志框架,只需要在我们工程中创建一个StaticLoggerBinder类,当然包名一定要是 org/slf4j/impl/

原文链接:https://blog.csdn.net/zhengwangzw/article/details/116036625


推荐阅读
  • spring boot使用jetty无法启动 ... [详细]
  • 本文探讨了如何通过Service Locator模式来简化和优化在B/S架构中的服务命名访问,特别是对于需要频繁访问的服务,如JNDI和XMLNS。该模式通过缓存机制减少了重复查找的成本,并提供了对多种服务的统一访问接口。 ... [详细]
  • 二维码的实现与应用
    本文介绍了二维码的基本概念、分类及其优缺点,并详细描述了如何使用Java编程语言结合第三方库(如ZXing和qrcode.jar)来实现二维码的生成与解析。 ... [详细]
  • Beetl是一款先进的Java模板引擎,以其丰富的功能、直观的语法、卓越的性能和易于维护的特点著称。它不仅适用于高响应需求的大型网站,也适合功能复杂的CMS管理系统,提供了一种全新的模板开发体验。 ... [详细]
  • 问题描述现在,不管开发一个多大的系统(至少我现在的部门是这样的),都会带一个日志功能;在实际开发过程中 ... [详细]
  • 问题场景用Java进行web开发过程当中,当遇到很多很多个字段的实体时,最苦恼的莫过于编辑字段的查看和修改界面,发现2个页面存在很多重复信息,能不能写一遍?有没有轮子用都不如自己造。解决方式笔者根据自 ... [详细]
  • Web动态服务器Python基本实现
    Web动态服务器Python基本实现 ... [详细]
  • Android与JUnit集成测试实践
    本文探讨了如何在Android项目中集成JUnit进行单元测试,并详细介绍了修改AndroidManifest.xml文件以支持测试的方法。 ... [详细]
  • 使用TabActivity实现Android顶部选项卡功能
    本文介绍如何通过继承TabActivity来创建Android应用中的顶部选项卡。通过简单的步骤,您可以轻松地添加多个选项卡,并实现基本的界面切换功能。 ... [详细]
  • Java 中的十进制样式 getZeroDigit()方法,示例 ... [详细]
  • 一、Advice执行顺序二、Advice在同一个Aspect中三、Advice在不同的Aspect中一、Advice执行顺序如果多个Advice和同一个JointPoint连接& ... [详细]
  • 如何将955万数据表的17秒SQL查询优化至300毫秒
    本文详细介绍了通过优化SQL查询策略,成功将一张包含955万条记录的财务流水表的查询时间从17秒缩短至300毫秒的方法。文章不仅提供了具体的SQL优化技巧,还深入探讨了背后的数据库原理。 ... [详细]
  • 本文将详细介绍如何使用Java编程语言生成指定数量的不重复随机数,包括具体的实现方法和代码示例。适合初学者和有一定基础的开发者参考。 ... [详细]
  • 本文介绍了如何通过C#语言调用动态链接库(DLL)中的函数来实现IC卡的基本操作,包括初始化设备、设置密码模式、获取设备状态等,并详细展示了将TextBox中的数据写入IC卡的具体实现方法。 ... [详细]
  • importjava.io.*;importjava.util.*;publicclass五子棋游戏{staticintm1;staticintn1;staticfinalintS ... [详细]
author-avatar
Fxnananana
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有