springboot默认日志框架选择源码解析(推荐)

作者:dengyouhua 时间:2021-06-19 08:02:51 

背景:

今天新生成一个springboot项目,然而启动日志,还有mybatis的详细日志无法打印出来,自写程序中打印的日志可以输出;网上找了很多资料,都没法解决问题;于是决定跟一下源码,弄清springboot日志相关的逻辑。

环境配置:macbook; intellij idea community edition 2020.03 ; gradle 6.8.3 jdk1.8 ;

gradle引用包如下:


dependencies {
 compile "com.alibaba:fastjson:1.2.75"
 compile "mysql:mysql-connector-java"

//spring
 compile("org.springframework.boot:spring-boot-starter")
 compile("org.mybatis.spring.boot:mybatis-spring-boot-starter:2.1.4")
 compile("org.springframework.boot:spring-boot-starter-web")
 compile("org.springframework.boot:spring-boot-starter-actuator")

//lombok
 compileOnly 'org.projectlombok:lombok'
 annotationProcessor 'org.projectlombok:lombok'

//test
 testCompile('org.springframework.boot:spring-boot-starter-test')
 testImplementation 'io.projectreactor:reactor-test'
}

springboot 默认日志使用的是logback(引入spring-boot-starter包后,就自动引入了logback-core,logback-classic两个包,当然还有slf4j的包),当springboot启动时,org.springframework.boot.context.logging.LoggingApplicationListener,该类211行注册的监控事件会被ApplicationStartingEvent触发;如下代码所示,会调用onApplicationStartingEvent初始化loggingSystem,而使用哪个日志组件,就要看loggingSystem初始化的值了


@Override
 public void onApplicationEvent(ApplicationEvent event) {
   if (event instanceof ApplicationStartingEvent) {
     onApplicationStartingEvent((ApplicationStartingEvent) event);
   }
   else if (event instanceof ApplicationEnvironmentPreparedEvent) {
     onApplicationEnvironmentPreparedEvent((ApplicationEnvironmentPreparedEvent) event);
   }
   else if (event instanceof ApplicationPreparedEvent) {
     onApplicationPreparedEvent((ApplicationPreparedEvent) event);
   }
   else if (event instanceof ContextClosedEvent
       && ((ContextClosedEvent) event).getApplicationContext().getParent() == null) {
     onContextClosedEvent();
   }
   else if (event instanceof ApplicationFailedEvent) {
     onApplicationFailedEvent();
   }
 }

private void onApplicationStartingEvent(ApplicationStartingEvent event) {
   this.loggingSystem = LoggingSystem.get(event.getSpringApplication().getClassLoader());
   this.loggingSystem.beforeInitialize();
 }

如下图是org.springframework.boot.logging.LoggingSystem类里面get函数的内容:首先会从system.getProperty中获取className,新生成的项目,取到的这个值都为空,SYSTEM_PROPERTY是一个固定值,就是该类的名字;那么loggingSystem的值就是从SYSTEM_FACTORY.getLoggingSystem(classLoader);获取到的;接下来我们得看LoggingSystemFactory.fromSpringFactories.getLoggingSystem取的值是什么了;


public static final String SYSTEM_PROPERTY = LoggingSystem.class.getName();
private static final LoggingSystemFactory SYSTEM_FACTORY = LoggingSystemFactory.fromSpringFactories();
public static LoggingSystem get(ClassLoader classLoader) {
   String loggingSystemClassName = System.getProperty(SYSTEM_PROPERTY);
   if (StringUtils.hasLength(loggingSystemClassName)) {
     if (NONE.equals(loggingSystemClassName)) {
       return new NoOpLoggingSystem();
     }
     return get(classLoader, loggingSystemClassName);
   }
   LoggingSystem loggingSystem = SYSTEM_FACTORY.getLoggingSystem(classLoader);
   Assert.state(loggingSystem != null, "No suitable logging system located");
   return loggingSystem;
 }

private static LoggingSystem get(ClassLoader classLoader, String loggingSystemClassName) {
   try {
     Class<?> systemClass = ClassUtils.forName(loggingSystemClassName, classLoader);
     Constructor<?> constructor = systemClass.getDeclaredConstructor(ClassLoader.class);
     constructor.setAccessible(true);
     return (LoggingSystem) constructor.newInstance(classLoader);
   }
   catch (Exception ex) {
     throw new IllegalStateException(ex);
   }
 }

LoggingSystemFactory是一个接口,它的实现类在spring-boot-start有4个,其中3个是在内部内类实现,DelegatingLoggingSystemFactory(JavaLoggingSystem,Log4J2LoggingSystem,LogbackLoggingSystem,内部类实现)。上面SYSTEM_FACTORY的实现就是DelegatingLoggingSystemFactory这个类,如下代码中delegates的值为JavaLoggingSystem,Log4J2LoggingSystem,LogbackLoggingSystem;三个类具体的加载逻辑在SpringFactoriesLoader.loadFactories函数中,最终返回的loggingSystem就是前面函数返回列表中的第一个;SpringFactoriesLoader.loadFactories 才是决定springboot默认会使用哪个日志组件关键:该类是spring的核心组件类,在spring-core包中,org.springframework.core.io.support.SpringFactoriesLoader;loggingSystem的值=LogbackLoggingSystem


public interface LoggingSystemFactory {

/**
 * Return a logging system implementation or {@code null} if no logging system is
 * available.
 * @param classLoader the class loader to use
 * @return a logging system
 */
 LoggingSystem getLoggingSystem(ClassLoader classLoader);

/**
 * Return a {@link LoggingSystemFactory} backed by {@code spring.factories}.
 * @return a {@link LoggingSystemFactory} instance
 */
 static LoggingSystemFactory fromSpringFactories() {
  return new DelegatingLoggingSystemFactory(
     (classLoader) -> SpringFactoriesLoader.loadFactories(LoggingSystemFactory.class, classLoader));
 }

}
class DelegatingLoggingSystemFactory implements LoggingSystemFactory {

private final Function<ClassLoader, List<LoggingSystemFactory>> delegates;

/**
 * Create a new {@link DelegatingLoggingSystemFactory} instance.
 * @param delegates a function that provides the delegates
 */
 DelegatingLoggingSystemFactory(Function<ClassLoader, List<LoggingSystemFactory>> delegates) {
  this.delegates = delegates;
 }

@Override
 public LoggingSystem getLoggingSystem(ClassLoader classLoader) {
  List<LoggingSystemFactory> delegates = (this.delegates != null) ? this.delegates.apply(classLoader) : null;
  if (delegates != null) {
    for (LoggingSystemFactory delegate : delegates) {
     LoggingSystem loggingSystem = delegate.getLoggingSystem(classLoader);
     if (loggingSystem != null) {
       return loggingSystem;
     }
    }
  }
  return null;
 }

}

总结:虽然springboot会加载JavaLoggingSystem,Log4J2LoggingSystem,LogbackLoggingSystem三个日志实现类,但在选择时,还是会使用LogbackLoggingSystem作为它的日志框架

来源:https://www.cnblogs.com/deng-1024/p/14525895.html

标签:springboot,日志框架
0
投稿

猜你喜欢

  • Java中char[]输出不是内存地址的原因详解

    2022-08-04 11:00:11
  • java开发Dubbo注解Adaptive实现原理

    2022-02-06 11:40:31
  • spring中的BeanFactory与FactoryBean的讲解

    2023-03-13 11:18:09
  • mybatis中foreach嵌套if标签方式

    2023-11-20 23:11:05
  • Spring Bean的线程安全问题

    2023-06-07 17:15:36
  • 关于Java双大括号{{}}的具体使用

    2021-05-28 16:32:01
  • Kotlin编程基础语法编码规范

    2023-06-20 16:50:07
  • c#多线程之线程基础

    2022-10-06 20:34:44
  • java实现Dijkstra最短路径算法

    2022-11-30 21:02:15
  • C#中读写INI文件的方法例子

    2022-10-21 23:14:48
  • Java由浅入深刨析继承

    2023-11-23 08:04:24
  • Java递归运行的机制:递归的微观解读图文分析

    2022-09-24 20:04:09
  • C#实现控制Windows系统关机、重启和注销的方法

    2023-07-24 17:15:05
  • android通过usb读取U盘的方法

    2023-03-14 07:43:25
  • Android Activity 不能被截屏的解决方法

    2021-06-28 08:37:35
  • 详解java为什么不允许类多重继承却允许接口多重继承

    2022-12-26 03:25:57
  • c#图像截取实例

    2022-02-21 07:49:58
  • jvm调优的几种场景(小结)

    2023-04-11 18:37:04
  • javassist使用指南

    2023-03-04 17:03:00
  • 解决idea删除模块后重新创建显示该模块已经被注册的问题

    2023-04-24 18:06:12
  • asp之家 软件编程 m.aspxhome.com