Spring Boot 源码分析系列02 logging模块日志组件装配

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,依赖关系如下:


image.png

具体分析下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。该类即为日志组件初始化的入口:


image.png
// 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子类的初始化方法。

image.png

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);

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351