SpringBoot之浅析配置项解析(二)

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 我们在上一篇文章中简单的说了一些SpringBoot配置属性相关的一些内容,我们在这篇文章中接着上一篇的文章继续进行分析。

我们在上一篇文章中简单的说了一些SpringBoot配置属性相关的一些内容,我们在这篇文章中接着上一篇的文章继续进行分析。我们在上一篇文章中提到了这样一个类:ConfigFileApplicationListener,从类名来看的话这是一个配置文件应用监听器,这个类主要的一个作用是在 refresh context之前解析默认的配置文件。首先我们来看一下它的UML类图:
ConfigFileApplicationListener
ConfigFileApplicationListener实现了EnvironmentPostProcessor和SmartApplicationListener这两个接口,从上图中我们可以看到SmartApplicationListener其实是继承了ApplicationListener这个接口的。SmartApplicationListener这个类相比于ApplicationListener多了两个方法,一个是用来检测是否支持给定的类型,一个是用来检测得到的source type。EnvironmentPostProcessor这个接口的作用是在refresh application context之前定制化应用的环境配置信息,需要说明的是,实现这个接口的类必须要在spring.factories进行配置。大家可以想一下这里为什么要这样。既然ConfigFileApplicationListener也是一个监听器类,那么它肯定会监听某些动作的发生,我们在之前的文章中说过,在org.springframework.boot.SpringApplication#run(java.lang.String… args)这个方法中会负责SpringBoot的整个启动工作,在这里有这样的一段代码:

        SpringApplicationRunListeners listeners = getRunListeners(args);
        listeners.starting();

顺着listeners.starting();这个方法往下扒的话,你就会找到调用org.springframework.boot.context.config.ConfigFileApplicationListener#onApplicationEvent这个方法的地方,这个调用链不算很负责,这里就不再多说了。我们直接进入到ConfigFileApplicationListener#onApplicationEvent这个方法中:

    public void onApplicationEvent(ApplicationEvent event) {
        if (event instanceof ApplicationEnvironmentPreparedEvent) {
            onApplicationEnvironmentPreparedEvent(
                    (ApplicationEnvironmentPreparedEvent) event);
        }
        if (event instanceof ApplicationPreparedEvent) {
            onApplicationPreparedEvent(event);
        }
    }

但是啊但是,在onApplicationEvent中有两个条件判断,你通过listeners.starting()这个调用链一直跟踪到这里的话你会发现,这两个条件都不满足listeners.starting()的调用,那么是在什么地方触发这个监听事件的呢?在org.springframework.boot.SpringApplication#prepareEnvironment这个方法中,我们在上一篇文章中介绍了其中的一些代码,这里再看一段代码:

listeners.environmentPrepared(environment);

org.springframework.boot.SpringApplicationRunListeners#environmentPrepared

    public void environmentPrepared(ConfigurableEnvironment environment) {
    //这里的listeners是从spring.factories中得到的
        for (SpringApplicationRunListener listener : this.listeners) {
            listener.environmentPrepared(environment);
        }
    }

org.springframework.boot.context.event.EventPublishingRunListener#environmentPrepared

public void environmentPrepared(ConfigurableEnvironment environment) {
//这里的ApplicationEvent是ApplicationEnvironmentPreparedEvent满足我们上面的提到的两个条件中的一个     this.initialMulticaster.multicastEvent(new ApplicationEnvironmentPreparedEvent(
                this.application, this.args, environment));
    }

从上面的代码中我们可以看到在执行listeners.environmentPrepared(environment);
的时候,会调用ConfigFileApplicationListener#onApplicationEvent这个方法,并且所传入的ApplicationEvent是ApplicationEnvironmentPreparedEvent,所以这里会调用ConfigFileApplicationListener#onApplicationEvent中的org.springframework.boot.context.config.ConfigFileApplicationListener#onApplicationEnvironmentPreparedEvent方法。我们进入到onApplicationEnvironmentPreparedEvent方法中看一下:

private void onApplicationEnvironmentPreparedEvent(
            ApplicationEnvironmentPreparedEvent event) {
        //从spring.factories中获取key为org.springframework.boot.env.EnvironmentPostProcessor的配置信息不再细说    
        List<EnvironmentPostProcessor> postProcessors = loadPostProcessors();
        //前面说过ConfigFileApplicationListener也实现了EnvironmentPostProcessor接口
        postProcessors.add(this);
        //排序
        AnnotationAwareOrderComparator.sort(postProcessors);
        //循环上一步得到的List,分别调用各自的postProcessEnvironment方法,在这里我们只关注ConfigFileApplicationListener这个类的postProcessEnvironment方法即可
        for (EnvironmentPostProcessor postProcessor : postProcessors) {
            postProcessor.postProcessEnvironment(event.getEnvironment(),
                    event.getSpringApplication());
        }
    }

postProcessEnvironment的源码如下:

    public void postProcessEnvironment(ConfigurableEnvironment environment,
            SpringApplication application) {
        //添加PropertySource 这个是我们要分析的重点
        addPropertySources(environment, application.getResourceLoader());
        //JavaBean内省的配置
        configureIgnoreBeanInfo(environment);
        bindToSpringApplication(environment, application);
    }

在上面的代码中,我们目前需要重点关注的就是ConfigFileApplicationListener#addPropertySources这个方法:

    protected void addPropertySources(ConfigurableEnvironment environment,
            ResourceLoader resourceLoader) {
//添加RandomValuePropertySource
RandomValuePropertySource.addToEnvironment(environment);
//加载默认配置信息,这个environment就是我们在SpringApplication中创建的StandardServletEnvironment
new Loader(environment, resourceLoader).load();
    }

RandomValuePropertySource.addToEnvironment

public static void addToEnvironment(ConfigurableEnvironment environment) {
    //MutablePropertySources#addAfter 这个方法我们在上一章中说过了这里就不再多说了。这段代码的意思是将RandomValuePropertySource的顺序添加到systemEnvironment的后面
    environment.getPropertySources().addAfter(
    //systemEnvironment
StandardEnvironment.SYSTEM_ENVIRONMENT_PROPERTY_SOURCE_NAME,
//random
        new RandomValuePropertySource(RANDOM_PROPERTY_SOURCE_NAME));
}

上面这段代码的意思就将RandomValuePropertySource添加到systemEnvironment的后面,到目前为止我们的MutablePropertySources#propertySourceList中的元素顺序为:commandLineArgs、servletConfigInitParams 、servletContextInitParams 、jndiProperties 、systemProperties 、systemEnvironment、random。我们接着分析new Loader(environment, resourceLoader).load();这段代码,这个代码中的内容比较多,我们也不是全部都要关注,这里我们先把Profile相关的内容先剥离出去。我们直接到这里:

            while (!this.profiles.isEmpty()) {
                Profile profile = this.profiles.poll();
                //获取配置文件的位置 1)
                for (String location : getSearchLocations()) {
                    //如果不是以/结尾的话则调用下面的方法
                    if (!location.endsWith("/")) {
                        // location is a filename already, so don't search for more
                        // filenames
                        load(location, null, profile);
                    }
                    else {
                        //获取配置文件的名称 2)
                        for (String name : getSearchNames()) {
                            load(location, name, profile);
                        }
                    }
                }
                this.processedProfiles.add(profile);
            }

我们先来看上面标记为1)处的代码:

        private Set<String> getSearchLocations() {
            Set<String> locations = new LinkedHashSet<String>();
            // User-configured settings take precedence, so we do them first
            //如果配置了spring.config.location这个属性值,则先解析这个属性值
            if (this.environment.containsProperty(CONFIG_LOCATION_PROPERTY)) {
                //多个值以 , 进行分割
                for (String path : asResolvedSet(
                        this.environment.getProperty(CONFIG_LOCATION_PROPERTY), null)) {
                    if (!path.contains("$")) {
                        path = StringUtils.cleanPath(path);
                        if (!ResourceUtils.isUrl(path)) {
                            path = ResourceUtils.FILE_URL_PREFIX + path;
                        }
                    }
                    locations.add(path);
                }
            }
            //通常我们都不会设置  ' spring.config.location ' 这个属性值 这里添加默认位置
            //默认位置为 "classpath:/,classpath:/config/,file:./,file:./config/"
            //asResolvedSet会用 , 分割上述字符串 并且还有一个作用将上传得到的List进行 翻转 注意这个很重要!!! 即使你在上面设置了 ' spring.config.location '  它的位置也是在系统默认的位置的后面
            locations.addAll(
                    asResolvedSet(ConfigFileApplicationListener.this.searchLocations,
                            DEFAULT_SEARCH_LOCATIONS));
            return locations;
        }

我们接着看2)处的代码:

        private Set<String> getSearchNames() {
        //如果配置了 spring.config.name 属性值
            if (this.environment.containsProperty(CONFIG_NAME_PROPERTY)) {
                return asResolvedSet(this.environment.getProperty(CONFIG_NAME_PROPERTY),
                        null);
            }
            //解析默认的文件名 application  这里同样会对获取到的List进行翻转
            return asResolvedSet(ConfigFileApplicationListener.this.names, DEFAULT_NAMES);
        }

所以上面while的这一段代码我们可以简化为这样(极其简化):

        List<String> listString = (Arrays.asList("file:./config/,file:./,classpath:/config/,classpath:/".split(",")));
        List<String> listName = Arrays.asList("application");
        for (String location : listString) {
            for(String name : listName){
                load(location, name, profile);
            }
        }
相关文章
|
3月前
|
并行计算 Java 数据处理
SpringBoot高级并发实践:自定义线程池与@Async异步调用深度解析
SpringBoot高级并发实践:自定义线程池与@Async异步调用深度解析
284 0
|
3月前
|
人工智能 自然语言处理 前端开发
SpringBoot + 通义千问 + 自定义React组件:支持EventStream数据解析的技术实践
【10月更文挑战第7天】在现代Web开发中,集成多种技术栈以实现复杂的功能需求已成为常态。本文将详细介绍如何使用SpringBoot作为后端框架,结合阿里巴巴的通义千问(一个强大的自然语言处理服务),并通过自定义React组件来支持服务器发送事件(SSE, Server-Sent Events)的EventStream数据解析。这一组合不仅能够实现高效的实时通信,还能利用AI技术提升用户体验。
276 2
|
2月前
|
前端开发 Java Maven
深入解析:如何用 Spring Boot 实现分页和排序
深入解析:如何用 Spring Boot 实现分页和排序
73 2
|
4月前
|
存储 缓存 Java
在Spring Boot中使用缓存的技术解析
通过利用Spring Boot中的缓存支持,开发者可以轻松地实现高效和可扩展的缓存策略,进而提升应用的性能和用户体验。Spring Boot的声明式缓存抽象和对多种缓存技术的支持,使得集成和使用缓存变得前所未有的简单。无论是在开发新应用还是优化现有应用,合理地使用缓存都是提高性能的有效手段。
61 1
|
3月前
|
前端开发 JavaScript Java
【SpringBoot系列】视图解析器的搭建与开发
【SpringBoot系列】视图解析器的搭建与开发
54 0
|
3月前
|
JavaScript 安全 Java
如何使用 Spring Boot 和 Ant Design Pro Vue 实现动态路由和菜单功能,快速搭建前后端分离的应用框架
本文介绍了如何使用 Spring Boot 和 Ant Design Pro Vue 实现动态路由和菜单功能,快速搭建前后端分离的应用框架。首先,确保开发环境已安装必要的工具,然后创建并配置 Spring Boot 项目,包括添加依赖和配置 Spring Security。接着,创建后端 API 和前端项目,配置动态路由和菜单。最后,运行项目并分享实践心得,包括版本兼容性、安全性、性能调优等方面。
210 1
|
2月前
|
JavaScript 安全 Java
如何使用 Spring Boot 和 Ant Design Pro Vue 构建一个具有动态路由和菜单功能的前后端分离应用。
本文介绍了如何使用 Spring Boot 和 Ant Design Pro Vue 构建一个具有动态路由和菜单功能的前后端分离应用。首先,创建并配置 Spring Boot 项目,实现后端 API;然后,使用 Ant Design Pro Vue 创建前端项目,配置动态路由和菜单。通过具体案例,展示了如何快速搭建高效、易维护的项目框架。
141 62
|
29天前
|
存储 JavaScript 前端开发
基于 SpringBoot 和 Vue 开发校园点餐订餐外卖跑腿Java源码
一个非常实用的校园外卖系统,基于 SpringBoot 和 Vue 的开发。这一系统源于黑马的外卖案例项目 经过站长的进一步改进和优化,提供了更丰富的功能和更高的可用性。 这个项目的架构设计非常有趣。虽然它采用了SpringBoot和Vue的组合,但并不是一个完全分离的项目。 前端视图通过JS的方式引入了Vue和Element UI,既能利用Vue的快速开发优势,
114 13
|
1月前
|
JavaScript 安全 Java
java版药品不良反应智能监测系统源码,采用SpringBoot、Vue、MySQL技术开发
基于B/S架构,采用Java、SpringBoot、Vue、MySQL等技术自主研发的ADR智能监测系统,适用于三甲医院,支持二次开发。该系统能自动监测全院患者药物不良反应,通过移动端和PC端实时反馈,提升用药安全。系统涵盖规则管理、监测报告、系统管理三大模块,确保精准、高效地处理ADR事件。
|
2月前
|
JavaScript 安全 Java
如何使用 Spring Boot 和 Ant Design Pro Vue 构建一个前后端分离的应用框架,实现动态路由和菜单功能
本文介绍了如何使用 Spring Boot 和 Ant Design Pro Vue 构建一个前后端分离的应用框架,实现动态路由和菜单功能。首先,确保开发环境已安装必要的工具,然后创建并配置 Spring Boot 项目,包括添加依赖和配置 Spring Security。接着,创建后端 API 和前端项目,配置动态路由和菜单。最后,运行项目并分享实践心得,帮助开发者提高开发效率和应用的可维护性。
140 2

推荐镜像

更多