01.日志使用背景
1.1日志使用
spring boot 的日志使用很简单,直接在工程目录的resource目录下创建一个logback-spring.xml就能配置日志格式了。我们大胆猜测一下,spring默认使用的是logback日志组件,并且帮我们自动装配了它。
1.2 spring boot 依赖管理
再来看一个小知识点,spring-boot-starter-parent和spring-boot-dependencies。在创建spring boot 应用的时候,一般的做法是在应用的maven配置文件中,添加对spring-boot-starter-parent的依赖。
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.1.0.RELEASE</version>
<relativePath/>
</parent>
这是因为spring帮我们统一管理了企业级应用常用的依赖。
仔细观察spring-boot-starter-parent的依赖,他的父级项目又是spring-boot-dependencies。
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-dependencies</artifactId>
<version>2.1.0.RELEASE</version>
<relativePath>../../spring-boot-dependencies</relativePath>
</parent>
<artifactId>spring-boot-starter-parent</artifactId>
在spring-boot-dependencies里面才真正定义了相关的依赖和版本。应用显示指定一下即可。
看到这里,其实应用直接依赖spring-boot-starter-parent作为父项目依赖的使用姿势是不对的,由于maven不支持多继承特性。所以正确的使用姿势是自定义一个maven pom工程,暂且命名为cli-dependencies,让这个工程依赖spring-boot-dependencies,所有的应用在依赖这个父项目。这么做的原因是方便统一管理其他spring-boot不包含的依赖版本。
<name>cli-dependencies</name>
<url>http://www.example.com</url>
<description>通用的依赖管理</description>
<!-- 在这里依赖sping boot 版本 -->
<!-- spring version @link https://start.spring.io/actuator/info -->
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-dependencies</artifactId>
<version>2.1.5.RELEASE</version>
<relativePath/>
</parent>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
<java.version>1.8</java.version>
<spring-cloud.version>Greenwich.SR5</spring-cloud.version>
</properties>
<dependencyManagement>
<dependencies>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-dependencies</artifactId>
<version>${spring-cloud.version}</version>
<type>pom</type>
<scope>import</scope>
</dependency>
</dependencies>
<!-- 在这里拓展可能需要用到的其他非spring-boot包含的jar,例如上面的spring-cloud -->
</dependencyManagement>
02.日志组件装配
2.1 分析依赖
spring-boot处理日志的模块命名为:spring-boot-starter-logging,依赖关系如下:
具体分析下logging模块:
<dependencies>
// 依赖logback
<dependency>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-classic</artifactId>
<version>1.2.3</version>
<scope>compile</scope>
</dependency>
// log4j在设计之初未考虑到使用sl4j作为门面
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-to-slf4j</artifactId>
<version>2.11.2</version>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>jul-to-slf4j</artifactId>
<version>1.7.26</version>
<scope>compile</scope>
</dependency>
</dependencies>
说明spring boot默认使用的日志组件就是logback,你想用其他组件,就得排除这个。另外也说明,如果要用log4j,可以直接使用,因为logging模块已经处理了sl4j门面兼容的问题。
2.2 组件初始化
springboot自动装配机制会读取spring-boot-2.1.5.RELEASE.jar!\META-INF\spring.factories文件内容,该文件配置了SpringBoot启动中要加载的组件。其中listeners中配置了org.springframework.boot.context.logging.LoggingApplicationListener。该类即为日志组件初始化的入口:
// LoggingApplicationListener.class
@Override
public void onApplicationEvent(ApplicationEvent event) {
// SpringApplication的run方法执行的时候触发该事件
if (event instanceof ApplicationStartedEvent) {
// onApplicationStartedEvent方法内部会先得到LoggingSystem,然后调用beforeInitialize方法
onApplicationStartedEvent((ApplicationStartedEvent) event);
}
// 环境信息准备好,ApplicationContext创建之前触发该事件
else if (event instanceof ApplicationEnvironmentPreparedEvent) {
// onApplicationEnvironmentPreparedEvent方法内部会做一下几个事情
// 1. 读取配置文件中"logging."开头的配置,比如logging.pattern.level, logging.pattern.console等设置到系统属性中
// 2. 构造一个LogFile(LogFile是对日志对外输出文件的封装),使用LogFile的静态方法get构造,会使用配置文件中logging.file和logging.path配置构造
// 3. 判断配置中是否配置了debug并为true,如果是,设置level的DEBUG,然后继续查看是否配置了trace并为true,如果是,设置level的TRACE
// 4. 构造LoggingInitializationContext,查看是否配置了logging.config,如有配置,调用LoggingSystem的initialize方法并带上该参数,否则调用initialize方法并且configLocation为null
// 5. 设置一些比如org.springframework.boot、org.springframework、org.apache.tomcat、org.apache.catalina、org.eclipse.jetty、org.hibernate.tool.hbm2ddl、org.hibernate.SQL这些包的log level,跟第3步的level一样
// 6. 查看是否配置了logging.register-shutdown-hook,如配置并设置为true,使用addShutdownHook的addShutdownHook方法加入LoggingSystem的getShutdownHandler
onApplicationEnvironmentPreparedEvent(
(ApplicationEnvironmentPreparedEvent) event);
}
// Spring容器创建好,并进行了部分操作之后触发该事件
else if (event instanceof ApplicationPreparedEvent) {
// onApplicationPreparedEvent方法内部会把LoggingSystem注册到BeanFactory中(前期是BeanFactory中不存在name为springBootLoggingSystem的实例)
onApplicationPreparedEvent((ApplicationPreparedEvent) event);
}
// Spring容器关闭的时候触发该事件
else if (event instanceof ContextClosedEvent && ((ContextClosedEvent) event)
.getApplicationContext().getParent() == null) {
// onContextClosedEvent方法内部调用LoggingSystem的cleanUp方法进行清除工作
onContextClosedEvent();
}
}
LoggingApplicationListener#onApplicationStartingEvent
LoggingSystem类很重要。它相当于一个工厂类,通过get方法去生成不同日志组件的实例。
LoggingSystem#get:
public static LoggingSystem get(ClassLoader classLoader) {
// 取系统参数org.springframework.boot.logging.LoggingSystem
String loggingSystem = System.getProperty(SYSTEM_PROPERTY);
if (StringUtils.hasLength(loggingSystem)) {
if (NONE.equals(loggingSystem)) {
return new NoOpLoggingSystem();
}
return get(classLoader, loggingSystem);
}
// 从SYSTEMS变量取出,并且判断类路径下是否有相关类【非常重要】
return SYSTEMS.entrySet().stream()
.filter((entry) -> ClassUtils.isPresent(entry.getKey(), classLoader))
.map((entry) -> get(classLoader, entry.getValue())).findFirst()
.orElseThrow(() -> new IllegalStateException(
"No suitable logging system located"));
}
所以,如果在程序启动时,不指定参数-Dorg.springframework.boot.logging.LoggingSystem的,会默认从集合SYSTEMS取第一个。因为LogbackLoggingSystem是第一个实现组件,所以会被默认使用。
private static final Map<String, String> SYSTEMS;
static {
// 有序的map集合
Map<String, String> systems = new LinkedHashMap<>();
systems.put("ch.qos.logback.core.Appender",
"org.springframework.boot.logging.logback.LogbackLoggingSystem");
systems.put("org.apache.logging.log4j.core.impl.Log4jContextFactory",
"org.springframework.boot.logging.log4j2.Log4J2LoggingSystem");
systems.put("java.util.logging.LogManager",
"org.springframework.boot.logging.java.JavaLoggingSystem");
SYSTEMS = Collections.unmodifiableMap(systems);
}
到这里基本就很清楚spring是如何加载bean组件了。
- LoggingApplicationListener被springboot自动装配,监听容器启动事件onApplicationStartingEvent;
- 通过LoggingSystem判断日志应该加载哪一个日志组件,优先从
org.springframework.boot.logging.LoggingSystem
系统参数中获取; - step2取不到再从静态变量
SYSTEMS
-有序的内建map。key是每种日志组件关键的实现类,value是spring-boot-logging-starter提供的实现。 - 循环遍历map,过滤类路径下是否存在。如果同时存在多个,则返回第一个。所以如果想用log4j,必须先要排除logback模块的依赖。否则永远取的是logback,因为这个map有序(logback排在前面)。
- 返回抽象的loggingSystem具体子类
2.3日志的初始化
调用run方法的时候,会从类路径或者系统参数找到需要装配的日志组件,在spring容器运行之前,开始调用Loggingsystem的初始化。LoggingApplicationListener#onApplicationEnvironmentPreparedEvent
,这里就会调用具体LoggingSystem子类的初始化方法。
LoggingSystem是个抽象类,内部有这几个方法:
- beforeInitialize方法:日志系统初始化之前需要处理的事情。抽象方法,不同的日志架构进行不同的处理
- initialize方法:初始化日志系统。默认不进行任何处理,需子类进行初始化工作
- cleanUp方法:日志系统的清除工作。默认不进行任何处理,需子类进行清除工作
- getShutdownHandler方法:返回一个Runnable用于当jvm退出的时候处理日志系统关闭后需要进行的操作,默认返回null,也就是什么都不做
- setLogLevel方法:抽象方法,用于设置对应logger的级别
以LogbackLoggingSystem为例,看下具体的初始化方法
// 标准的配置文件的名称
@Override
protected String[] getStandardConfigLocations() {
return new String[] { "logback-test.groovy", "logback-test.xml", "logback.groovy",
"logback.xml" };
}
// 初始化
@Override
public void initialize(LoggingInitializationContext initializationContext,
String configLocation, LogFile logFile) {
LoggerContext loggerContext = getLoggerContext();
if (isAlreadyInitialized(loggerContext)) {
return;
}
// 先调用AbstractLoggingSystem#initialize
super.initialize(initializationContext, configLocation, logFile);
loggerContext.getTurboFilterList().remove(FILTER);
markAsInitialized(loggerContext);
if (StringUtils.hasText(System.getProperty(CONFIGURATION_FILE_PROPERTY))) {
getLogger(LogbackLoggingSystem.class.getName()).warn(
"Ignoring '" + CONFIGURATION_FILE_PROPERTY + "' system property. "
+ "Please use 'logging.config' instead.");
}
}
AbstractLoggingSystem
@Override
public void initialize(LoggingInitializationContext initializationContext,
String configLocation, LogFile logFile) {
// 如果有配置具体的文件,从指定文件加载 logging.config=classpath:**.xml
if (StringUtils.hasLength(configLocation)) {
initializeWithSpecificConfig(initializationContext, configLocation, logFile);
return;
}
// 没有就去resource目录下找,下面看下找的策略
initializeWithConventions(initializationContext, logFile);
}
......
// 具体查找配置文件的策略
private void initializeWithConventions(
LoggingInitializationContext initializationContext, LogFile logFile) {
// 获取日志组件默认的配置
// 01 getStandardConfigLocations获取内建的4个配置文件名称
//02 判断任意一个存在即返回
String config = getSelfInitializationConfig();
if (config != null && logFile == null) {
// self initialization has occurred, reinitialize in case of property changes
reinitialize(initializationContext);
return;
}
// 如果上面4个文件都找不到,查找spring扩展的4个文件,扩展策略类似于 logback.xml->logback-spring.xml,判断classpath路径是否存在。
if (config == null) {
config = getSpringInitializationConfig();
}
// 配置不为空 直接初始化
if (config != null) {
loadConfiguration(initializationContext, config, logFile);
return;
}
// 配置文件为空,加载默认配置
loadDefaults(initializationContext, logFile);
}
小结:
1.日志配置文件叫logback.xml或者logback-spring.xml都可以。
2.日志组件装配的思想和spring boot大致相同。利用条件装配,判断类路径下是否有具体的类存在。ClassUtils.isPresent(entry.getKey(), classLoader)
3.日志配置文件的获取策略也是判断resource目录下的指定名称的配置文件是否存在。``
参考:ClassPathResource resource = new ClassPathResource(location, this.classLoader);