java sl4j 日志_Java日志框架:slf4j作用及其实现原理

简单回顾门面模式

slf4j是门面模式的典型应用,因此在讲slf4j前,我们先简单回顾一下门面模式,

门面模式,其核心为外部与一个子系统的通信必须通过一个统一的外观对象进行,使得子系统更易于使用。用一张图来表示门面模式的结构为:

门面模式的核心为Facade即门面对象,门面对象核心为几个点:

知道所有子角色的功能和责任

将客户端发来的请求委派到子系统中,没有实际业务逻辑

不参与子系统内业务逻辑的实现

大致上来看,对门面模式的回顾到这里就可以了,开始接下来对SLF4J的学习。

我们为什么要使用slf4j

我们为什么要使用slf4j,举个例子:

我们自己的系统中使用了logback这个日志系统

我们的系统使用了A.jar,A.jar中使用的日志系统为log4j

我们的系统又使用了B.jar,B.jar中使用的日志系统为slf4j-simple

这样,我们的系统就不得不同时支持并维护logback、log4j、slf4j-simple三种日志框架,非常不便。

解决这个问题的方式就是引入一个适配层,由适配层决定使用哪一种日志系统,而调用端只需要做的事情就是打印日志而不需要关心如何打印日志,slf4j或者commons-logging就是这种适配层,slf4j是本文研究的对象。

从上面的描述,我们必须清楚地知道一点:slf4j只是一个日志标准,并不是日志系统的具体实现。理解这句话非常重要,slf4j只做两件事情:

提供日志接口

提供获取具体日志对象的方法

slf4j-simple、logback都是slf4j的具体实现,log4j并不直接实现slf4j,但是有专门的一层桥接slf4j-log4j12来实现slf4j。

为了更理解slf4j,我们先看例子,再读源码,相信读者朋友会对slf4j有更深刻的认识。

slf4j应用举例

上面讲了,slf4j的直接/间接实现有slf4j-simple、logback、slf4j-log4j12,我们先定义一个pom.xml,引入相关jar包:

1

2

3 xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">

4 4.0.0

5

6 org.xrq.log

7 log-test

8 1.0.0

9 jar

10

11 log-test

12 http://maven.apache.org

13

14

15 UTF-8

16

17

18

19

20 junit

21 junit

22 4.11

23 test

24

25

26 org.slf4j

27 slf4j-api

28 1.7.25

29

30

31 ch.qos.logback

32 logback-classic

33 1.2.3

34

35

36 org.slf4j

37 slf4j-simple

38 1.7.25

39

40

41 log4j

42 log4j

43 1.2.17

44

45

46 org.slf4j

47 slf4j-log4j12

48 1.7.21

49

50

51

写一段简单的Java代码:

1 @Test2 public voidtestSlf4j() {3 Logger logger = LoggerFactory.getLogger(Object.class);4 logger.error("123");5 }

接着我们首先把上面pom.xml的第30行~第49行注释掉,即不引入任何slf4j的实现类,运行Test方法,我们看一下控制台的输出为:

看到没有任何日志的输出,这验证了我们的观点:slf4j不提供日志的具体实现,只有slf4j是无法打印日志的。

接着打开logback-classic的注释,运行Test方法,我们看一下控制台的输出为:

看到我们只要引入了一个slf4j的具体实现类,即可使用该日志框架输出日志。

最后做一个测验,我们把所有日志打开,引入logback-classic、slf4j-simple、log4j,运行Test方法,控制台输出为:

和上面的差别是,可以输出日志,但是会输出一些告警日志,提示我们同时引入了多个slf4j的实现,然后选择其中的一个作为我们使用的日志系统。

从例子我们可以得出一个重要的结论,即slf4j的作用:只要所有代码都使用门面对象slf4j,我们就不需要关心其具体实现,最终所有地方使用一种具体实现即可,更换、维护都非常方便。

slf4j实现原理

上面看了slf4j的示例,下面研究一下slf4j的实现,我们只关注重点代码。

slf4j的用法就是常年不变的一句"Logger logger = LoggerFactory.getLogger(Object.class);",可见这里就是通过LoggerFactory去拿slf4j提供的一个Logger接口的具体实现而已,LoggerFactory的getLogger的方法实现为:

1 public static Logger getLogger(Class>clazz) {2 Logger logger =getLogger(clazz.getName());3 if(DETECT_LOGGER_NAME_MISMATCH) {4 Class> autoComputedCallingClass =Util.getCallingClass();5 if (autoComputedCallingClass != null &&nonMatchingClasses(clazz, autoComputedCallingClass)) {6 Util.report(String.format("Detected logger name mismatch. Given name: \"%s\"; computed name: \"%s\".", logger.getName(),7 autoComputedCallingClass.getName()));8 Util.report("See " + LOGGER_NAME_MISMATCH_URL + " for an explanation");9 }10 }11 returnlogger;12 }

从第2行开始跟代码,一直跟到LoggerFactory的bind()方法:

1 private final static voidbind() {2 try{3 Set staticLoggerBinderPathSet = null;4 //skip check under android, see also5 // http://jira.qos.ch/browse/SLF4J-328

6 if (!isAndroid()) {7 staticLoggerBinderPathSet =findPossibleStaticLoggerBinderPathSet();8 reportMultipleBindingAmbiguity(staticLoggerBinderPathSet);9 }10 //the next line does the binding

11 StaticLoggerBinder.getSingleton();12 INITIALIZATION_STATE =SUCCESSFUL_INITIALIZATION;13 reportActualBinding(staticLoggerBinderPathSet);14 fixSubstituteLoggers();15 replayEvents();16 //release all resources in SUBST_FACTORY

17 SUBST_FACTORY.clear();18 } catch(NoClassDefFoundError ncde) {19 String msg =ncde.getMessage();20 if(messageContainsOrgSlf4jImplStaticLoggerBinder(msg)) {21 INITIALIZATION_STATE =NOP_FALLBACK_INITIALIZATION;22 Util.report("Failed to load class \"org.slf4j.impl.StaticLoggerBinder\".");23 Util.report("Defaulting to no-operation (NOP) logger implementation");24 Util.report("See " + NO_STATICLOGGERBINDER_URL + " for further details.");25 } else{26 failedBinding(ncde);27 throwncde;28 }29 } catch(java.lang.NoSuchMethodError nsme) {30 String msg =nsme.getMessage();31 if (msg != null && msg.contains("org.slf4j.impl.StaticLoggerBinder.getSingleton()")) {32 INITIALIZATION_STATE =FAILED_INITIALIZATION;33 Util.report("slf4j-api 1.6.x (or later) is incompatible with this binding.");34 Util.report("Your binding is version 1.5.5 or earlier.");35 Util.report("Upgrade your binding to version 1.6.x.");36 }37 thrownsme;38 } catch(Exception e) {39 failedBinding(e);40 throw new IllegalStateException("Unexpected initialization failure", e);41 }42 }

这个地方第7行是一个关键,看一下代码:

1 static SetfindPossibleStaticLoggerBinderPathSet() {2 //use Set instead of list in order to deal with bug #1383 //LinkedHashSet appropriate here because it preserves insertion order4 //during iteration

5 Set staticLoggerBinderPathSet = new LinkedHashSet();6 try{7 ClassLoader loggerFactoryClassLoader = LoggerFactory.class.getClassLoader();8 Enumerationpaths;9 if (loggerFactoryClassLoader == null) {10 paths =ClassLoader.getSystemResources(STATIC_LOGGER_BINDER_PATH);11 } else{12 paths =loggerFactoryClassLoader.getResources(STATIC_LOGGER_BINDER_PATH);13 }14 while(paths.hasMoreElements()) {15 URL path =paths.nextElement();16 staticLoggerBinderPathSet.add(path);17 }18 } catch(IOException ioe) {19 Util.report("Error getting resources from path", ioe);20 }21 returnstaticLoggerBinderPathSet;22 }

这个地方重点其实就是第12行的代码,getLogger的时候会去classpath下找STATIC_LOGGER_BINDER_PATH,STATIC_LOGGER_BINDER_PATH值为"org/slf4j/impl/StaticLoggerBinder.class",即所有slf4j的实现,在提供的jar包路径下,一定是有"org/slf4j/impl/StaticLoggerBinder.class"存在的,我们可以看一下:

我们不能避免在系统中同时引入多个slf4j的实现,所以接收的地方是一个Set。大家应该注意到,上部分在演示同时引入logback、slf4j-simple、log4j的时候会有警告:

这就是因为有三个"org/slf4j/impl/StaticLoggerBinder.class"存在的原因,此时reportMultipleBindingAmbiguity方法控制台输出语句:

1 private static void reportMultipleBindingAmbiguity(SetbinderPathSet) {2 if(isAmbiguousStaticLoggerBinderPathSet(binderPathSet)) {3 Util.report("Class path contains multiple SLF4J bindings.");4 for(URL path : binderPathSet) {5 Util.report("Found binding in [" + path + "]");6 }7 Util.report("See " + MULTIPLE_BINDINGS_URL + " for an explanation.");8 }9 }

那网友朋友可能会问,同时存在三个"org/slf4j/impl/StaticLoggerBinder.class"怎么办?首先确定的是这不会导致启动报错,其次在这种情况下编译期间,编译器会选择其中一个StaticLoggerBinder.class进行绑定,这个地方sfl4j也在reportActualBinding方法中报告了绑定的是哪个日志框架:

1 private static void reportActualBinding(SetbinderPathSet) {2 //binderPathSet can be null under Android

3 if (binderPathSet != null &&isAmbiguousStaticLoggerBinderPathSet(binderPathSet)) {4 Util.report("Actual binding is of type [" + StaticLoggerBinder.getSingleton().getLoggerFactoryClassStr() + "]");5 }6 }

对照上面的截图,看最后一行,确实是"Actual binding is of type..."这句。

最后StaticLoggerBinder就比较简单了,不同的StaticLoggerBinder其getLoggerFactory实现不同,拿到ILoggerFactory之后调用一下getLogger即拿到了具体的Logger,可以使用Logger进行日志输出。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 这个问题通常是由于缺少 slf4j 相关的 jar 包引起的。slf4j 是一个日志框架,它提供了一种通用的日志接口,可以与多种日志系统(如 log4j、logback 等)进行适配。而 StaticLoggerBinder 是 slf4j 的一个默认实现,它需要在 classpath 中引入 slf4j 的 jar 包,并且这些 jar 包的版本需要兼容。你可以检查一下你的项目中是否引入了正确版本的 slf4j 相关的 jar 包,并且这些 jar 包是否与其他依赖的 jar 包兼容。 ### 回答2: java.lang.ClassNotFoundException: org.slf4j.impl.StaticLoggerBinder是一个Java异常,表示无法找到org.slf4j.impl.StaticLoggerBinder类。 在Java中,Slf4j(Simple Logging Facade for Java)是一种日志记录框架,它提供了一个简单的日志接口,并允许开发人员在需要的时候切换具体的日志实现。StaticLoggerBinder是Slf4j框架的一部分,它是Slf4j与具体的日志实现(如logback、log4j等)之间的桥梁。 当我们在程序执行过程中遇到java.lang.ClassNotFoundException: org.slf4j.impl.StaticLoggerBinder异常时,意味着JVM无法找到并加载org.slf4j.impl.StaticLoggerBinder类,主要有以下几个可能原因: 1. 缺少Slf4j的实现库:这意味着你的项目依赖包中没有Slf4j的具体实现库,如logback或log4j。你需要添加正确的Slf4j实现库到你的classpath中。 2. 依赖冲突:当项目中存在多个Slf4j的实现库时,可能会引发依赖冲突。你需要确保项目中只有一个Slf4j的实现库,避免版本冲突。 3. 类路径配置错误:检查类路径配置是否正确,确保org.slf4j.impl.StaticLoggerBinder类所在的jar包在类路径中。 4. 部署环境问题:如果是在特定的部署环境中出现该异常,可能是由于环境配置问题。你可以尝试重新配置环境,确保Sl4j的实现库正确加载。 解决该异常的方法包括: 1. 添加Slf4j的具体实现库:将正确的Slf4j实现库添加到项目的依赖中。 2. 解决依赖冲突:检查项目的依赖关系,确保只有一个Slf4j的实现库。 3. 检查类路径配置:确认类路径配置是否正确,包括Slf4j的实现库所在的jar包。 4. 检查部署环境:如果该异常在特定的部署环境中出现,检查环境配置是否正确。 总之,要解决java.lang.ClassNotFoundException: org.slf4j.impl.StaticLoggerBinder异常,需要确认Slf4j的实现库正确添加到项目中,并排除依赖冲突、类路径配置错误和部署环境问题。 ### 回答3: java.lang.ClassNotFoundException: org.slf4j.impl.StaticLoggerBinder 是一个常见的错误,它通常出现在使用Slf4j日志框架时,无法找到 StaticLoggerBinder 类的情况下。这个问题通常是由于缺少相关的Slf4j依赖库或配置问题导致的。 要解决该问题,首先要确保在项目的构建路径中包含正确的Slf4j依赖库。可以通过在项目的构建文件(如pom.xml)中添加如下依赖来引入Slf4j: ```xml <dependency> <groupId>org.slf4j</groupId> <artifactId>slf4j-api</artifactId> <version>1.7.32</version> </dependency> <dependency> <groupId>org.slf4j</groupId> <artifactId>slf4j-simple</artifactId> <version>1.7.32</version> </dependency> ``` 这两个依赖分别是Slf4j的API和简单实现。确保版本号与您使用的Slf4j版本一致。 另外,也可能是由于在项目的类路径中存在多个版本的Slf4j库导致的冲突。可以通过检查项目的依赖关系,排除掉重复的Slf4j库来解决冲突问题。 此外,也可以检查项目中的Slf4j配置文件是否正确。Slf4j默认会在类路径中查找名为 "simplelogger.properties" 的配置文件。确保该文件存在并正确配置Slf4j的相关属性,例如日志输出级别等。 综上所述,当出现 java.lang.ClassNotFoundException: org.slf4j.impl.StaticLoggerBinder 错误时,我们需要检查Slf4j的依赖库是否正确引入,解决依赖冲突问题,并确保Slf4j配置文件的正确性,以此来解决该错误。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值