spring boot启动源码分析(三)之Environment准备

news/2025/1/10 17:07:59/

上一篇《spring-boot启动源码分析(二)之SpringApplicationRunListener》

环境介绍:

spring boot版本:2.7.18

主要starter:spring-boot-starter-web

本篇开始讲启动过程中Environment环境准备,Environment是管理所有配置的实例对象,像application.yml、系统属性、环境变量等配置都可以通过Environment.getProperty(key)获取

入口如下:

(一)创建ConfigurableEnvironment

首先会从spring.factories中获取ApplicationContextFactory.class接口实现类,根据webApplicationType会实例化对应的ConfigurableEnvironment(这里是SERVLET,所以是AnnotationConfigServletWebServerApplicationContext.Factory创建的ApplicationServletEnvironment)

ApplicationServletEnvironment的类结构如下:

ApplicationServletEnvironment调用构造方法实例化时,父类AbstractEnvironment构造方法中会有初始化操作:

propertySources是Environment的核心属性,包含了各个配置源,这里赋值了一个MutablePropertySources,它实际是一个迭代器:

propertyResolver:是属性解析器,如占位符的解析等

AbstractEnvironment构造器采用类似模板方法将customizePropertySources(propertySources)交给子类实现,子类可以在此方法中加载数据源

可以看到子类StandardServletEnvironment,添加了StubPropertySource,桩配置源,类似打桩,这两个桩配置源分别是servletConfigInitParams和servletContextInitParams。这两个在webserver启动后会被替换成实际的配置源,这里只是个站位,没有实际的作用。

之后再次调用超类的customizePropertySources:

在StandardEnvironment中会添加两个重要的配置源,systemProperties和systemEnvironment。

systemProperties对应System.getProperties(),即所有的系统属性:

systemEnvironment对应的是System.getenv(),即系统的所有环境变量

(二)对Environment进行其他配置

     

        (1)首先添加conversionService:environment中的propertyResolver(配置解析器)配置应用转换服务,例如NumberToNumberConverterFactory,StringToCharacterConverter等。应该是为propertyResource中的数据解析时进行数据转换用的

        (2)configurePropertySources:

        如果SpringApplication中Map<String, Object> defaultProperties不为空,则会创建一个名为“defaultProperties”,实例类型为DefaultPropertiesPropertySource的配置源。

        java进程传入的命令行参数加载进配置源中,配置源名称为commandLineArgs,实例类型为CommandLinePropertySource。此方法为模板方法,可被子类覆盖实现自己的资源加载方式configureProfiles

        (3)此为空实现,看注释是说配置此应用程序环境中哪些配置文件处于活动状态(或默认情况下处于活动状态)。在配置文件处理过程中,可以通过{@code-spring.profiles.active}属性激活其他配置文件。

(三)绑定ConfigurationPropertySources

ConfigurationPropertySources.attach(environment);

增加一个名为“configurationProperties”的ConfigurationPropertySourcesPropertySource和environment绑定,实际上是把environment中所有的propertySource(也包括configurationProperties)包装进一个SpringConfigurationPropertySources实例中,而SpringConfigurationPropertySources是ConfigurationPropertySourcesPropertySource中的一个属性值,这也意味着configurationProperties可以访问所有的propertySource,这样它可以作为配置的统一访问入口。可以看如下configurationProperties对应实例属性

所以ConfigurationPropertySourcesPropertySource.getProperty(String name),实际上就是遍历每个PropertySource,获取它们的配置,但它对每个PropertySource进行了适配,以便以统一的接口进行获取配置。

getSource获取的是SpringConfigurationPropertySources,是一个迭代器,iterator()中会对配置源进行适配,适配成ConfigurationPropertySource

(四)发布environmentPrepared事件

事件的发布流程都差不多,这里不再赘述。主要差别是哪些监听器会监听了此事件,并做了什么逻辑处理。这里我们重点说一下这个:

主要有6个监听器:

代理监听器可以跳过,没有实际的逻辑处理。我们关注spring boot自定义的监听器。

(1)EnvironmentPostProcessorApplicationListener

这里主要是从spring.fatories获取EnvironmentPostProcessor接口实现类并实例化:

然后调用对应postProcessEnvironment,这些EnvironmentPostProcessor主要是添加不同的数据源配置,如ConfigDataEnvironmentPostProcessor解析我们常用的application.yml作为配置源,详情可看《Spring boot源码之EnvironmentPostProcessor》

(2)AnsiOutputApplicationListener

这里主要是根据spring.output.ansi.enabled和spring.output.ansi.console-available,设置AnsiOutput对应的属性。为输出到控制台的日志信息添加 ANSI 转义码,以实现彩色输出。这个用的比较少,我看只在打印banner和logback日志的颜色转换器中用到了

(3)LoggingApplicationListener

在上一篇发布starting事件中,日志系统进行了初步初始化,在这里则会完成全部的初始化。

a、getLoggingSystemProperties(environment).apply():会将配置文件中配置的日志相关配置设置到系统属性中,以便后续日志系统初始化时可以从系统属性读取。

如:配置文件中如果设置了logging.pattern.console,那么就会设置到系统属性变量CONSOLE_LOG_PATTERN中。这里还设置了PID以及下面的日志策略

b、initializeEarlyLoggingLevel(environment):初始化早期日志级别,实际只是根据Environment中是否配置debug和trace,设置springBootLogging对应级别

c、initializeSystem(environment, this.loggingSystem, this.logFile):这里是主要的初始化逻辑,如果之前日志已经初始化过,这里会使用spring boot会重新初始化,对日志进行增强。比如在logback中,会使用SpringBootJoranConfigurator,对配置文件进行加载解析,同时新定义了一些解析规则

所以在spring boot中可以使用springProperty,添加属性,值可以是Environment中配置的属性值

如上,是配置了一个pid变量,值是Environment中pid属性对应的属性值

d、initializeFinalLoggingLevels(environment, this.loggingSystem):初始化日志的最终的日志级别:

如果b步骤设置了springBootLogging,那么spring boot会初始化一些logger,设置其日志级别,如过是debug级别,则下图圈出来的都会设置是debug级别

另外则是对配置属性中有logging.level为前缀的设置其对应的日志级别:

如上,则会设置com.example为true。

e、registerShutdownHookIfNecessary(environment, this.loggingSystem)

往SpringApplication注册日志系统的shutdownHandler,会在shutdown发生时,停止日志上下文。

(4)BackgroundPreinitializer

启动一个后台线程去出发早期的初始化,并且有个countDownLatch,只有初始化完成,当ApplicationReadyEvent监听触发时,得等线程执行完才会继续执行,否则等待

(5)FileEncodingApplicationListener

        当系统设置了spring.mandatory-file-encoding编码格式时,会判断是否和System.getProperty("file.encoding")相同,如果不相等则抛出异常阻止系统的启动,默认没设置不影响

(五)其他

上面基本已经把环境变量准备完毕,剩下的是一些细微处理:

(1)将defaultProperties配置源移到最后,即查询资源时最后才查它

(2)SpringApplication相关环境变量绑定

即将spring.main开头的配置,绑定到对应SpringApplication的属性中。如spring.main.allow-circular-references=true,那么就会设置SpringApplication的allowCircularReferences值为true.

(3)Environment转换ConfigurableEnvironment为应用类型的环境,这里是ApplicationServletEnvironment,是ConfigurableEnvironment的子类,不需要转换


http://www.ppmy.cn/news/1562026.html

相关文章

DFS之剪枝

常用减枝顺序&#xff1a; 优化搜索顺序&#xff1a;大部分情况下&#xff0c;我们应该优先搜索分支较少的结点排除等效冗余可行性剪枝最优性剪枝最优化搜索&#xff08;DP&#xff09; 165. 小猫爬山 翰翰和达达饲养了 N N N 只小猫&#xff0c;这天&#xff0c;小猫们要去…

《零基础Go语言算法实战》 【题目 1-15】字符串的比较

《零基础Go语言算法实战》 【题目 1-15】字符串的比较 请用 Go 语言实现一个算法&#xff0c;在不使用额外存储结构的条件下判断一个字符串的所有字 符是否全都相同&#xff0c;字符串的长度不能超过 3000。 【解答】 ① 思路。 本题需要实现一个算法来判断字符串中的所有…

【System Verilog and UVM基础入门25】功能覆盖率的实现

tdc_coverage.sv 让代码开口说话!!! 记住,代码是最可靠的朋友,它永远不会说谎! `uvm_analysis_imp_decl(_in_monitor_export) `uvm_analysis_imp_decl(_out_monitor_export)class tdc_coverage extends uvm_component;`uvm_component_utils(tdc_coverage)tdc_config m…

ESP32 IDF VScode出现头文件“无法打开 源 文件 ”,并有红色下划线警告

问题背景&#xff1a; ESP32 IDF VScode出现头文件“无法打开 源 文件 ”&#xff0c;并有红色下划线警告&#xff1a; 解决办法&#xff1a; 在工程里面的.vscode文件夹下&#xff0c;检查是否存在c_cpp_properties.json文件&#xff0c;如果没有可以手动创建添加。如图…

升级 Spring Boot 3 配置讲解 — 新版本的秒杀系统怎么做?

学会这款 &#x1f525;全新设计的 Java 脚手架 &#xff0c;从此面试不再怕&#xff01; 1. Spring Boot 3 升级指南 在升级 Spring Boot 3 之前&#xff0c;首先需要确保你的项目已经升级到 Java 17&#xff0c;因为 Spring Boot 3 不再支持 Java 8 和 Java 11。接下来&…

文件读写到SQLite数据库的方法

在 SQLite 数据库中&#xff0c;将文件读写到数据库的常见方法主要有以下几种&#xff1a; 1. 将文件以 BLOB 类型存储 BLOB&#xff08;Binary Large Object&#xff09; 是 SQLite 中的二进制数据类型&#xff0c;可以直接用来存储文件内容。 步骤&#xff1a; 创建表 创建一…

HTML+CSS+JS制作中国传统节日主题网站(内附源码,含5个页面)

一、作品介绍 HTMLCSSJS制作一个中国传统节日主题网站&#xff0c;包含首页、节日介绍页、民俗文化页、节日活动页、联系我们页等5个静态页面。其中每个页面都包含一个导航栏、一个主要区域和一个底部区域。 二、页面结构 1. 顶部横幅区 包含传统中国风格的网站标题中国传统…

解决Vscode中使用netdb.h的getaddrinfo和addrinfo会报错的方法

原文地址&#xff1a;https://kashima19960.github.io/2024/12/03/解决Vscode中使用netdb.h的getaddrinfo和addrinfo会报错的方法/&#xff0c;一般有最新的修改都是在我的个人博客里面&#xff0c;所以在当前平台的更新会比较慢&#xff0c;请见谅&#x1f603; 前言 博主最近…