【Spring Boot 源码学习】深入应用上下文初始化器实现

《Spring Boot 源码学习系列》

在这里插入图片描述

深入应用上下文初始化器实现

  • 一、引言
  • 二、往期内容
  • 三、主要内容
    • 3.1 spring-boot 子模块中内置的实现类
      • 3.1.1 ConfigurationWarningsApplicationContextInitializer
      • 3.1.2 ContextIdApplicationContextInitializer
      • 3.1.3 DelegatingApplicationContextInitializer
      • 3.1.4 RSocketPortInfoApplicationContextInitializer
      • 3.1.5 ServerPortInfoApplicationContextInitializer
    • 3.2 spring-boot-autoconfigure 子模块中内置的实现类
      • 3.2.1 SharedMetadataReaderFactoryContextInitializer
      • 3.2.2 ConditionEvaluationReportLoggingListener
    • 3.3 自定义应用上下文初始化器实现
      • 3.3.1 定义 DemoApplicationContextInitializer
      • 3.3.2 添加 DemoApplicationContextInitializer
      • 3.3.3 实际演示
  • 四、总结

一、引言

前面的博文《ApplicationContextInitializer 详解》,Huazie 带大家详细分析了 ApplicationContextInitializer 的加载和初始化的逻辑,不过有关 ApplicationContextInitializer 接口的实现尚未提及 。那本篇 Huazie 就带大家一起分析 Spring Boot 中预置的应用上下文初始化器实现【即 ApplicationContextInitializer 接口实现类】的源码,了解在 Spring 容器刷新之前初始化应用程序上下文的一些具体操作。

在这里插入图片描述

二、往期内容

在开始本篇的内容介绍之前,我们先来看看往期的系列文章【有需要的朋友,欢迎关注系列专栏】:

Spring Boot 源码学习
Spring Boot 项目介绍
Spring Boot 核心运行原理介绍
【Spring Boot 源码学习】@EnableAutoConfiguration 注解
【Spring Boot 源码学习】@SpringBootApplication 注解
【Spring Boot 源码学习】走近 AutoConfigurationImportSelector
【Spring Boot 源码学习】自动装配流程源码解析(上)
【Spring Boot 源码学习】自动装配流程源码解析(下)
【Spring Boot 源码学习】深入 FilteringSpringBootCondition
【Spring Boot 源码学习】OnClassCondition 详解
【Spring Boot 源码学习】OnBeanCondition 详解
【Spring Boot 源码学习】OnWebApplicationCondition 详解
【Spring Boot 源码学习】@Conditional 条件注解
【Spring Boot 源码学习】HttpEncodingAutoConfiguration 详解
【Spring Boot 源码学习】RedisAutoConfiguration 详解
【Spring Boot 源码学习】JedisConnectionConfiguration 详解
【Spring Boot 源码学习】初识 SpringApplication
【Spring Boot 源码学习】Banner 信息打印流程
【Spring Boot 源码学习】自定义 Banner 信息打印
【Spring Boot 源码学习】BootstrapRegistryInitializer 详解
【Spring Boot 源码学习】ApplicationContextInitializer 详解
【Spring Boot 源码学习】ApplicationListener 详解
【Spring Boot 源码学习】SpringApplication 的定制化介绍
【Spring Boot 源码学习】BootstrapRegistry 详解
【Spring Boot 源码学习】深入 BootstrapContext 及其默认实现
【Spring Boot 源码学习】BootstrapRegistry 初始化器实现
【Spring Boot 源码学习】BootstrapContext的实际使用场景

三、主要内容

注意: 以下涉及 Spring Boot 源码 均来自版本 2.7.9,其他版本有所出入,可自行查看源码。

3.1 spring-boot 子模块中内置的实现类

我们先来看一张截图:

在这里插入图片描述

从上图中可以看出,spring-boot 子模块中配置的 ApplicationContextInitializer 实现一共有 5 个,下面我们一一来介绍下:

3.1.1 ConfigurationWarningsApplicationContextInitializer

该类用于报告常见配置错误的警告,我们来看看相关源码:

public class ConfigurationWarningsApplicationContextInitializer
		implements ApplicationContextInitializer<ConfigurableApplicationContext> {

	@Override
	public void initialize(ConfigurableApplicationContext context) {
		context.addBeanFactoryPostProcessor(new ConfigurationWarningsPostProcessor(getChecks()));
	}

	// 省略其他。。。
}

阅读上述源码,我们可以看到 initialize 方法里,通过 ConfigurableApplicationContextaddBeanFactoryPostProcessor 方法,在 应用程序上下文 中添加了一个 BeanFactoryPostProcessor 实现【 该实现类为 ConfigurationWarningsPostProcessor】。

BeanFactoryPostProcessorSpring 框架中的一个接口,它的作用是在 Spring 容器初始化时对 Bean 的定义进行修改或增强【添加属性、设置依赖关系等】。

在介绍 ConfigurationWarningsPostProcessor 之前,先来看看 getChecks 方法:

protected Check[] getChecks() {
	return new Check[] { new ComponentScanPackageCheck() };
}

我们继续查看 ComponentScanPackageCheck ,由于篇幅受限,Huazie 贴下截图:

在这里插入图片描述

ComponentScanPackageCheckConfigurationWarningsApplicationContextInitializer 中的一个静态内部类,它的目的是在Spring Boot 应用启动时,检查 @ComponentScan 的使用情况,确保没有错误或不推荐的配置方式。通过 ComponentScanPackageCheckgetWarning 方法的检查,如果发现了不恰当的 @ComponentScan 使用,它会生成相应的警告信息,帮助开发者及时发现并修正潜在的配置问题。

下面我们可以来分析下 ConfigurationWarningsPostProcessor,如下截图:

在这里插入图片描述

该类也是一个静态内部类,它同时实现了 PriorityOrderedBeanDefinitionRegistryPostProcessor 接口:

  • PriorityOrdered :实现该接口是用于提高其在多个 BeanFactoryPostProcessor 处理中的执行优先级。
  • BeanDefinitionRegistryPostProcessor:它是对 BeanFactoryPostProcessor 的扩展,允许在常规的 BeanFactoryPostProcessor 检测启动之前注册更多的 bean 定义,这些定义反过来定义了BeanFactoryPostProcessor 实例【可查看 PostProcessorRegistrationDelegate 了解】。

从上述截图中,我们可以看到 postProcessBeanFactory 方法【BeanFactoryPostProcessor 接口定义的方法】是空实现,而postProcessBeanDefinitionRegistry 方法【BeanDefinitionRegistryPostProcessor 接口定义的方法】里,遍历了 checks 数组中的每个检查项,并调用 check.getWarning(registry) 方法获取警告信息。如果警告信息不为空,则调用私有方法 warn(message) 打印警告信息。

3.1.2 ContextIdApplicationContextInitializer

先来看看 ContextIdApplicationContextInitializer 的源码,如下:

在这里插入图片描述
在这里插入图片描述

通过阅读上述源码,可以看出 ContextIdApplicationContextInitializer 是一个用于设置 Spring ApplicationContext ID 的应用上下文初始化器。其中,spring.application.name 属性用于创建 ID。如果该属性未设置,则使用 application

我们在 initialize 方法中,还看到了如下的代码:

applicationContext.getBeanFactory().registerSingleton(ContextId.class.getName(), contextId);

这里就是将一个名为 ContextId 的类注册为单例对象,并将其存储在 SpringApplicationContext 中。然后我们就可以在应用程序的不同部分共享和重用同一个 ContextId 实例,而无需每次都创建新的实例。

3.1.3 DelegatingApplicationContextInitializer

同样先来看看 DelegatingApplicationContextInitializer 的源码,如下截图:

在这里插入图片描述

通过阅读该类的 initialize 方法,我们可以看出 DelegatingApplicationContextInitializer 初始化工作是委托给其他在 context.initializer.classes 环境属性下指定的应用上下文初始化器进行的。

下面的 3.3 小节,我们在自定义 ApplicationContext 初始化器实现 时就会用到。

3.1.4 RSocketPortInfoApplicationContextInitializer

无需多言,直接查看 RSocketPortInfoApplicationContextInitializer 的源码,如下:

public class RSocketPortInfoApplicationContextInitializer
		implements ApplicationContextInitializer<ConfigurableApplicationContext> {
	@Override
	public void initialize(ConfigurableApplicationContext applicationContext) {
		applicationContext.addApplicationListener(new Listener(applicationContext));
	}
	
	// 省略。。。
}

阅读上述的 initialize 方法,可以看到这里向应用上下文中添加了一个 ApplicationListener,而这个 ListenerRSocketPortInfoApplicationContextInitializer 中的一个静态内部类。

继续阅读 Listener 的源码:

在这里插入图片描述

Listener 是用来监听 RSocketServerInitializedEvent 事件,该事件是在应用程序上下文刷新且 RSocketServer 准备就绪后发布的。

继续查看 onApplicationEvent 方法,我们可以看出该监听器是用来设置 RSocketServer 服务器实际监听的端口的环境属性。属性 local.rsocket.server.port 可以直接使用 @Value 注入到测试中,也可以通过 Environment 获取。另外该属性会自动向上传播到任何父上下文。

3.1.5 ServerPortInfoApplicationContextInitializer

同样还是从 ServerPortInfoApplicationContextInitializer 源码入手,如下所示:

在这里插入图片描述

通过阅读上面的 initialize 方法,可以看到这里也是比较简单,直接向应用上下文中添加了一个 ApplicationListener ,当然这个应用事件监听器比较特殊,就是其本身,因为 ServerPortInfoApplicationContextInitializer 实现了 ApplicationListener 接口。

ApplicationListener 监听的事件是 WebServerInitializedEvent,它是一个在 WebServer 准备就绪时发布的事件。

我们继续阅读 onApplicationEvent 方法的源码:

在这里插入图片描述

我们来简单总结如下:

该应用事件监听器用于设置 WebServer 服务器实际监听的端口的环境属性。属性 local.server.port 【如果 WebServerInitializedEvent 有一个服务器命名空间,它将被用来构造属性名称。例如,“management” actuator 上下文将具有属性名称 local.management.port】可以直接使用 @Value 注入到测试中,也可以通过 Environment 获取。该属性同样会自动向上传播到任何父上下文。

ActuatorSpring Boot 提供的一个开发库,它允许开发人员在运行时监控和管理应用程序。通过 Actuator,你可以查看应用程序的运行状况、性能指标、日志信息等。同时,它也提供了一些内置的管理端点,如健康检查、环境信息、应用信息等,方便开发人员进行调试和监控。Actuator 还提供了扩展机制,允许你自定义管理端点,以满足特定的需求。

3.2 spring-boot-autoconfigure 子模块中内置的实现类

同样我们先看截图:

在这里插入图片描述

从上图中可以看出,spring-boot-autoconfigure 子模块中配置的 ApplicationContextInitializer 实现有 2 个,下面来简单介绍下:

3.2.1 SharedMetadataReaderFactoryContextInitializer

SharedMetadataReaderFactoryContextInitializer 是一个应用上下文初始化器,主要作用是在 Spring 应用程序上下文创建之初,初始化一个共享的 MetadataReaderFactory 实例到在 Spring 应用上下文中。这样,在整个应用程序生命周期内,不同的组件在需要读取类的元数据时,都可以使用一个共享的 MetadataReaderFactory 实例,而无需每次都创建新的实例。

Spring 中,元数据(metadata)是用来描述 Bean 信息的数据,例如类名、方法名、属性名等。在应用程序运行时,Spring 会读取这些元数据来创建和管理 Bean。而 MetadataReaderFactory 就是负责读取和解析类的元数据,比如注解、类属性等。

这块的逻辑比较复杂,Huazie 后续将再出一篇博文详细分析,敬请期待!

3.2.2 ConditionEvaluationReportLoggingListener

ConditionEvaluationReportLoggingListener 是一个用于将 ConditionEvaluationReport 写入日志的应用上下文初始化器,该应用上下文初始化器并不打算在多个应用程序上下文实例之间共享。

Spring 应用程序上下文初始化时,它会评估所有使用条件注解的 bean 定义和配置。这些条件可能基于类是否存在、特定的属性设置、其他 bean 是否存在等。ConditionEvaluationReport 记录了每个条件注解的评估结果,包括哪些条件通过了(即 bean 或配置被创建或执行了),哪些条件没有通过(即 bean 或配置被跳过了)。

ConditionEvaluationReport 的评估结果报告默认将以 DEBUG 级别进行记录。崩溃报告会触发 info 级别的输出,建议再次运行并启用 debug 级别以显示报告。

这块的逻辑也比较复杂,Huazie 后续也会出一篇博文详细介绍下,大家可以期待一下。

3.3 自定义应用上下文初始化器实现

上面 Huazie 同大家一起分析了 Spring Boot 中一些内置的应用上下文初始化器实现,相信对于如何实现 ApplicationContextInitializer 接口,已经有了较为深入的了解。

3.3.1 定义 DemoApplicationContextInitializer

那下面就让我们自定义 ApplicationContext 初始化器实现,如下所示:

public class DemoApplicationContextInitializer implements ApplicationContextInitializer<ConfigurableApplicationContext>, Ordered {

    private int order = 0;

    @Override
    public void initialize(ConfigurableApplicationContext applicationContext) {
        User user = new User("Huazie", 18);
        applicationContext.getBeanFactory().registerSingleton(User.class.getName(), user);
    }

    public void setOrder(int order) {
        this.order = order;
    }

    @Override
    public int getOrder() {
        return this.order;
    }

}

上述 DemoApplicationContextInitializerinitialize 方法中,我们注册了一个 User 类的单例 Bean

3.3.2 添加 DemoApplicationContextInitializer

现在自定义的应用上下文初始化器有了,我们该如何添加它呢?

通过阅读 SpringApplication 的源码 和 本篇 3.1.3 小节的介绍,我们可以总结如下的三种方式:

  • META-INF/spring.factories 中添加 org.springframework.context.ApplicationContextInitializer 的配置。这种方式,我们从 《ApplicationContextInitializer 详解》 的 3.2 小节可见一斑。

    org.springframework.context.ApplicationContextInitializer=com.example.demo.DemoApplicationContextInitializer
    
  • 通过 SpringApplication 中的 addInitializers 方法添加。其实这里在笔者的《SpringApplication 的定制化介绍》中的 1.6 小节也提及过。

    SpringApplication springApplication = new SpringApplication(DemoApplication.class);
    springApplication.addInitializers(new DemoApplicationContextInitializer());
    // 其他省略。。。
    
  • application.properties 中添加 context.initializer.classes 的属性配置。这里实际上来源于 3.1.3 小节的 DelegatingApplicationContextInitializer

    # 逗号分隔的类名列表
    context.initializer.classes=com.example.demo.DemoApplicationContextInitializer
    

    application.yml 中添加 context.initializer.classes 的属性配置

    # 在 YAML 中,数组或列表元素使用 - 符号来定义
    context:  
    	initializer:  
    		classes:  
      			- com.example.demo.DemoApplicationContextInitializer  
    

3.3.3 实际演示

我们采用第三种添加方式,配置截图如下:

在这里插入图片描述

添加如下自测类【用来演示获取在 DemoApplicationContextInitializer 中注册的 User 类的单例 Bean 对象】:

import com.example.demo.pojo.User;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;

@SpringBootTest
public class DemoApplicationTests {

    @Autowired
    private User user;

    @Test
    public void test() {
        System.out.println("User = " + user);
    }
}

我们来看看运行结果,如下所示:

在这里插入图片描述
从上图可以看出,我们自定义的应用上下文初始化器实现显然已经执行了,并且成功注册了 User 类的单例 Bean 对象。

四、总结

本篇 Huazie 带大家一起分析了 Spring Boot 中预置的应用上下文初始化器实现,然后自定义了一个应用上下文初始化器实现类,进一步加深了对 Spring Boot 初始化应用上下文过程的了解,为后续的启动运行过程的理解打下了坚实的基础。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/465068.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

FFmpeg-aac、h264封装flv及时间转换

文章目录 时间概念流程api核心代码 时间概念 dts: 解码时间戳, 表示压缩帧的解码时间 pts: 显示时间戳, 表示将压缩帧解码后得到的原始帧的显示时间 时间基: time_base &#xff0c; 通常以ms为单位 时间戳: timestamp , 多少个时间基 真实时间&#xff1a;time_base * timest…

email + celery+django 异步发送邮件功能的实现

主要流程&#xff1a; django通过发件服务器到收件服务器&#xff0c;最后到收件人 邮件配置设置需要打开SMTP/IMAP并获的授权码&#xff0c;完成授权功能实现发送给收件人 邮件配置请参考另一博客https://blog.csdn.net/qq_44238024/article/details/136277821 项目结构树…

mac下Appuim环境安装

参考资料 Mac安装Appium_mac电脑安装appium-CSDN博客 安卓测试工具&#xff1a;Appium 环境安装&#xff08;mac版本&#xff09;_安卓自动化测试mac环境搭建-CSDN博客 1. 基本环境依赖 1 node.js 2 JDK&#xff08;Java JDK&#xff09; 3 Android SDK 4 Appium&#x…

数据分析 | Matplotlib

Matplotlib 是 Python 中常用的 2D 绘图库&#xff0c;它能轻松地将数据进行可视化&#xff0c;作出精美的图表。 绘制折线图&#xff1a; import matplotlib.pyplot as plt #时间 x[周一,周二,周三,周四,周五,周六,周日] #能量值 y[61,72,66,79,80,88,85] # 用来设置字体样式…

Linux进程管理:(六)SMP负载均衡

文章说明&#xff1a; Linux内核版本&#xff1a;5.0 架构&#xff1a;ARM64 参考资料及图片来源&#xff1a;《奔跑吧Linux内核》 Linux 5.0内核源码注释仓库地址&#xff1a; zhangzihengya/LinuxSourceCode_v5.0_study (github.com) 1. 前置知识 1.1 CPU管理位图 内核…

如何用Selenium通过Xpath,精准定位到“多个相同属性值以及多个相同元素”中的目标属性值

前言 本文是该专栏的第21篇,后面会持续分享python爬虫干货知识,记得关注。 相信很多同学,都有使用selenium来写爬虫项目或者自动化页面操作项目。同样,也相信很多同学在使用selenium来定位目标元素的时候,或多或少遇见到这样的情况,就是用Xpath定位目标元素的时候,页面…

Mysql主从之keepalive+MySQL高可用

一、Keepalived概述 keepalived 是集群管理中保证集群高可用的一个服务软件&#xff0c;用来防止单点故障。 keepalived 是以VRRP 协议为实现基础的&#xff0c;VRRP 全称VirtualRouter Redundancy Protocol&#xff0c;即虚拟路由冗余协议。虚拟路由冗余协议&#xff0c;可以…

launchctl及其配置、使用、示例

文章目录 launchctl 是什么Unix / Linux类似的工具有什么哪个更常用配置使用常用子命令示例加载一个 launch agent:卸载一个 launch daemon:列出所有已加载的服务:启动一个服务:停止一个服务:禁用一个服务:启用一个服务: 附com.example.myagent.plist内容有趣的例子参考 launch…

力扣L15--- 67.二进制求和(JAVA版)-2024年3月17日

1.题目描述 2.知识点 注1&#xff1a; 二进制用 %2 /2 3.思路和例子 采用竖位相加的方法 4.代码实现 class Solution {public String addBinary(String a, String b) {StringBuilder sbnew StringBuilder();int ia.length()-1;int jb.length()-1;int jinwei0;int digit1,d…

快速排序(数据结构)

1. 前言&#xff1a; 这两种排序经常使用&#xff0c;且在算法题中经常遇见。 这里我们简单分析讨论一下。 1. 快速排序 平均时间复杂度&#xff1a;O&#xff08;nlogn&#xff09; 最坏时间复杂度&#xff1a; O&#xff08;n^2&#xff09; 1.1. 左右向中遍历: 取最右侧4…

Multiplicity - 用一个键盘和鼠标控制多台电脑

Multiplicity 是一款用于多台电脑间控制的软件。通过这个工具&#xff0c;用户可以轻松地在多个计算机之间共享剪贴板、鼠标、键盘和显示屏幕。这样&#xff0c;无需每台电脑之间频繁切换&#xff0c;工作效率也会大大提高。 特征 远程PC访问 无缝控制过渡 兼容所有显示类型…

【Linux杂货铺】进程的基本概念

目录 &#x1f308;前言&#x1f308; &#x1f4c1;进程的概念 &#x1f4c2;描述进程-PCB &#x1f4c2; 查看进程 &#x1f4c2; 查看正在运行的程序 &#x1f4c2;杀死进程 &#x1f4c2;通过系统调用获取进程标识符 &#x1f4c2;通过系统调用创建进程 &#x1f…

万界星空科技商业开源MES,技术支持+项目合作

商业开源的一套超有价值的JAVA制造执行MES系统源码 亲测 带本地部署搭建教程 教你如何在本地运行运行起来。 开发环境&#xff1a;jdk11tomcatmysql8springbootmaven 可以免费使用&#xff0c;需要源码价格便宜&#xff0c;私信我获取。 一、系统概述&#xff1a; MES制造执…

机器学习(26)回顾gan+文献阅读

文章目录 摘要Abstract一、李宏毅机器学习——GAN1. Introduce1.1 Network as Generator1.2 Why distribution 2. Generative Adversarial Network2.1 Unconditional generation2.2 Basic idea of GAN 二、文献阅读1. 题目2. abstract3. 网络架构3.1 Theoretical Results 4. 文…

学习数据结构和算法的第16天

单链表的实现 链表的基本结构 #pragma once #include<stdio.h> #include<stlib.h> typedf int SLTDataType; typedy struct SListNode {SLTDataType data;struct SListNode*next; }SLTNode;void Slisprint(SLTNode*phead); void SListPushBack(SLTNode**pphead,S…

使用 VS Code + Github 搭建个人博客

搭建个人博客的方案 现在&#xff0c;搭建个人博客的方式有很多&#xff0c;门槛也很低。 可以选择已有平台&#xff1a; 掘金语雀知乎简书博客园SegmentFault… 也可以选择一些主流的博客框架&#xff0c;自行搭建。 HexoGitBookVuePressdumi… 如何选择&#xff1f; 我…

每日五道java面试题之mybatis篇(三)

目录&#xff1a; 第一题. MyBatis的框架架构设计是怎么样的?第二题. 为什么需要预编译?第三题. Mybatis都有哪些Executor执行器&#xff1f;它们之间的区别是什么&#xff1f;第四题. Mybatis中如何指定使用哪一种Executor执行器&#xff1f;第五题. Mybatis是否支持延迟加载…

如何学习一个大型分布式Java项目

前言 很多同学在没有实习经验的时候看到一个多模块分布式项目总是有一种老虎吃天的无力感&#xff0c;就像我刚毕业去到公司接触项目的时候一样&#xff0c;模块多的夸张&#xff0c;想学都不知道从哪开始学&#xff0c;那么我们拿到一份代码后如何从头开始学习一个新项目呢。…

挑战杯 机器视觉目标检测 - opencv 深度学习

文章目录 0 前言2 目标检测概念3 目标分类、定位、检测示例4 传统目标检测5 两类目标检测算法5.1 相关研究5.1.1 选择性搜索5.1.2 OverFeat 5.2 基于区域提名的方法5.2.1 R-CNN5.2.2 SPP-net5.2.3 Fast R-CNN 5.3 端到端的方法YOLOSSD 6 人体检测结果7 最后 0 前言 &#x1f5…

【鸿蒙HarmonyOS开发笔记】常用组件介绍篇 —— Button按钮组件

概述 Button为按钮组件&#xff0c;通常用于响应用户的点击操作。 参数 Button组件有两种使用方式&#xff0c;分别是不包含子组件和包含子组件&#xff0c;两种方式下&#xff0c;Button 组件所需的参数有所不同&#xff0c;下面分别介绍 不包含子组件 不包含子组件时&…