springboot默認日志框架選擇源碼解析(推薦)
背景:
今天新生成一個springboot項目,然而啟動日志,還有mybatis的詳細日志無法打印出來,自寫程序中打印的日志可以輸出;網(wǎng)上找了很多資料,都沒法解決問題;于是決定跟一下源碼,弄清springboot日志相關的邏輯。
環(huán)境配置: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行注冊的監(jiān)控事件會被ApplicationStartingEvent觸發(fā);如下代碼所示,會調(diào)用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函數(shù)的內(nèi)容:首先會從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是一個接口,它的實現(xiàn)類在spring-boot-start有4個,其中3個是在內(nèi)部內(nèi)類實現(xiàn),DelegatingLoggingSystemFactory(JavaLoggingSystem,Log4J2LoggingSystem,LogbackLoggingSystem,內(nèi)部類實現(xiàn))。上面SYSTEM_FACTORY的實現(xiàn)就是DelegatingLoggingSystemFactory這個類,如下代碼中delegates的值為JavaLoggingSystem,Log4J2LoggingSystem,LogbackLoggingSystem;三個類具體的加載邏輯在SpringFactoriesLoader.loadFactories函數(shù)中,最終返回的loggingSystem就是前面函數(shù)返回列表中的第一個;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; } }
總結(jié):雖然springboot會加載JavaLoggingSystem,Log4J2LoggingSystem,LogbackLoggingSystem三個日志實現(xiàn)類,但在選擇時,還是會使用LogbackLoggingSystem作為它的日志框架
到此這篇關于springboot默認日志框架選擇源碼解析的文章就介紹到這了,更多相關springboot日志框架內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Java 數(shù)據(jù)結(jié)構(gòu)進階二叉樹題集上
二叉樹可以簡單理解為對于一個節(jié)點來說,最多擁有一個上級節(jié)點,同時最多具備左右兩個下級節(jié)點的數(shù)據(jù)結(jié)構(gòu)。本文將帶你通過實際題目來熟練掌握2022-04-04servlet之session工作原理簡介_動力節(jié)點Java學院整理
這篇文章主要介紹了servlet之session工作原理簡介,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-07-07Sublime Text 打開Java文檔中文亂碼的解決方案
這篇文章主要介紹了Sublime Text 中文亂碼的解決方案,幫助大家更好的理解和使用Java,感興趣的朋友可以了解下2020-12-12java利用pdfbox+poi往pdf插入數(shù)據(jù)
這篇文章主要給大家介紹了關于java利用pdfbox+poi如何往pdf插入數(shù)據(jù)的相關資料,文中通過實例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2022-02-02Java中notify()和notifyAll()的使用區(qū)別
本文主要介紹了Java中notify()和notifyAll()的使用區(qū)別,文中通過示例代碼介紹的非常詳細,感興趣的小伙伴們可以參考一下2021-06-06