Spring关于@Configuration配置处理流程解析

news2024/12/24 2:30:44

@Configuration配置处理流程解析

  • AnnotationConfigApplicationContext基于注解配置
  • ApplicationContext启动刷新流程
  • Spring关于@Configuration解析处理流程
  • 那些年被忽略问题

AnnotationConfigApplicationContext基于注解配置

Spring通过上下文应用AnnotationConfigApplicationContext提供基于注解配置能力,启动过程中通过ConfigurationClassPostProcessor在执行Bean定义注册后处理器阶段解析@Configuration、@Bean、@Import等注解,完成Bean声明定义和注册到容器注册器中。本文简述Spring容器启动流程对配置类@Configuration处理过程,理解Spring应用如何实现基于@Configuration配置注册。

ApplicationContext启动刷新流程

Spring容器启动流程AbstractApplicationContext#refresh,列举几个关键阶段(感兴趣同学自行debug):

  1. obtainFreshBeanFactory,获取Bean工厂。
  2. prepareBeanFactory,准备工作,设置系统相关Bean定义等。
  3. postProcessBeanFactory,设置Bean工厂后处理。
  4. invokeBeanFactoryPostProcessors,执行Bean工厂后置处理器,按Bean定义注册器后处理,Bean工厂后置处理顺序执行。
  5. registerBeanPostProcessors,注册Bean后处理器。
  6. initMessageSource、initApplicationEventMulticaster,初始化国际化消息和Spring事件广播处理器。
  7. registerListeners,注册监听器。
  8. finishBeanFactoryInitialization,固化配置,完成单例非延迟初始化对象实例化。

Spring关于@Configuration解析处理流程

上面第4点,Spring可以通过自定义Bean定义后置处理,自定义解析Bean定义规则并完成Bean定义注册到Spring容器中,@Configuration就是通过bean定义注册后置处理器ConfigurationClassPostProcessor实现,关键处理步骤如下(感兴趣同学自行debug):

  1. ConfigurationClassPostProcessor#postProcessBeanDefinitionRegistry,Bean定义后置处理器执行入口。
  2. ConfigurationClassPostProcessor#processConfigBeanDefinitions,通过扫描应用已注册配置,引导扫描或导入其他候选配置类。
  3. ConfigurationClassParser#parse,解析每一个配置类,按如下顺序处理,具体代码见:
    ConfigurationClassParser#doProcessConfigurationClass
    1)、获取配置类内部定义的成员类,即静态内部类和实例内部类,如果内部成员类为配置类,则递归处理内部成员类。
    2)、处理配置类上面@PropertySource注解,属性文件注入处理。
    3)、扫描@ComponentScan @ComponentScans指定路径下Component类(ComponentScanAnnotationParser#parse),如果扫描类为配置类,则递归处理扫描到的类。
    4)、处理@Import,导入指定配置类,导入方式分ImportSelector,ImportBeanDefinitionRegistrar和processConfigurationClass。
    5)、处理@ImportResource,导入指定配置资源文件。
    6)、处理配置类内部定义的@Bean methods
    7)、处理default methods on interfaces
    8)、处理superClass
  4. ConfigurationClassBeanDefinitionReader#loadBeanDefinitions,加载3解析出来@Bean,对于静态方法Bean和实例方法Bean定义区别,需要关注ConfigurationClassBeanDefinitionReader#loadBeanDefinitionsForBeanMethod。

那些年被忽略问题

基于对配置类@Configuration处理流程理解,试下是否能正确回答下面两个问题:
问题 1:@Configuration配置类实例方法声明@Bean和静态方法声明@Bean有什么区别,对应用启动有产生什么影响?
样例代码如下:

@Configuration
public class UserConfiguration {
	@Bean
	public static UserService userService() {
		return new UserService();
	}
	/*@Bean
	public UserService userService() {
		return new UserService();
	}*/
}

@Data
public class UserService {
	public String sayHello() {
		return "hello";
	}
}

解析 1:从@Configuration解析处理流程可以知道,@Bean方法在ConfigurationClassBeanDefinitionReader#loadBeanDefinitions中通过包装成工厂方法去声明Bean定义,实例方法调用需要依赖于类实例对象,故会触发类实例化,而静态方法不需要依赖于实例对象,故不会触发类实例化。故如果是类实例方法声明@Bean,会导致工厂配置类过早实例化,从会引起不可预期异常。
关于这点,Spring官方明确说明:https://docs.spring.io/spring-framework/reference/core/beans/java/composing-configuration-classes.html
在这里插入图片描述

  1. @Configuration配置类依赖应该简单,因为配置类实例化可能会提前,依赖注入关系会导致非预期异常。
  2. @Configuration如果有定义bean工厂后处理器和bean后处理器,应该声明为静态@Bean方法,避免导致配置类过早实例化。

问题 2: @Configuration配置类内部实例类和内部静态类内声明Bean,Bean实例化顺序是什么?
样例代码如下:

@Configuration
public class UserOuterConfiguration {
	@Bean
	public UserService userService1() {
		return new UserService("userService1");
	}

	@Configuration
	public static class UserInnerConfiguration {
		@Bean
		public UserService userService2() {
			return new UserService("userService2");
		}
	}
}

@Data
@AllArgsConstructor
public static class UserService {
	private String name;
	
	public String sayHello() {
		return "hello";
	}
}

解析2:在组件扫描ComponentScan处理候选Configuration,会扫描指定包名下的配置类,按名称字典序排序。这个会让内部配置类优先与配置类处理,bean实例化顺序依赖于bean注册器里面注册的顺序,所以在不存在依赖关系,即没有如@Autoware @DependsOn @Lazy等会影响bean实例化顺序和Import导入配置类等改变某个配置处理顺序前提下,内部静态配置类定义Bean的实例化会优先于外部配置类定义Bean实例化。

by 三木

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

10- c语言复合数据类型 (C语言)

一 结构体 1.1 引入 1、在自然界中 任何一个物体,都有多个属性,如果用计算机语言来描述的话,一个属性也许可以用某一个基本数据类型来表示,但是当有多个属性的时候,一个基本数据类型就不能表示了。例如:学…

ML算法——线代预备知识随笔【机器学习】

文章目录 数学预备知识3、线性代数3.1、矩阵奇异值分解(SVD)3.2、广义逆矩阵(Moore-Penrose )3.3、数据白化(Data Whitening)3.4、向量导数 4、其它 数学预备知识 3、线性代数 3.1、矩阵奇异值分解&#…

最小化暗数据风险的 5 个步骤

超过一半的公司数据存储库包含哪些内容,但大多数人甚至不知道自己拥有什么?这是暗数据,是公司在不知不觉中收集的信息,它们不是日常业务交互的组成部分,因此通常位于后台。 虽然这些数据对于大多数公司来说似乎是不必…

C语言——数据的输入输出

数据的输入输出 前言:一、格式输入输出函数1.格式输出函数printf()2.格式输入函数scanf() 二、字符输入输出函数1.字符输出函数putchar()2.字符输入函数getchar() 三、字…

JUC并发工具类--阻塞队列BlockingQueue

JUC并发工具类--阻塞队列BlockingQueue 队列队列(Queue接口)提供的方法 阻塞队列阻塞队列(BlockingQueue接口)提供的方法应用场景JUC包下的阻塞队列如何选择适合的阻塞队列选择策略线程池对于阻塞队列的选择 队列 是限定在一端进…

Servlet技术实现服务端,Android平台作为客户端,实现一个个人店铺

背景: 使用Servlet技术实现服务端,使用Android平台作为客户端,实现一个个人店铺,店铺商品不限。功能要求如下: 1. 提供登录、注册功能;(10分) 2. 首页面包括“商品列表”子页面、“…

ATA-8000系列射频功率放大器——在生物医学中的应用

ATA-8000系列射频功率放大器——在生物医学研究中的应用 ATA-8000系列是一款射频功率放大器。其P1dB输出功率500W,饱和输出功率最大1000W。增益数控可调,一键保存设置,提供了方便简洁的操作选择,可与主流的信号发生器配套使用&…

VulnHub靶机渗透:SKYTOWER: 1

SKYTOWER: 1 靶机环境介绍nmap扫描端口扫描服务扫描漏洞扫描总结 80端口目录爆破 3128端口获取立足点获取立足点2提权总结 靶机环境介绍 https://www.vulnhub.com/entry/skytower-1,96/ 靶机IP:192.168.56.101 kali IP:192.168.56.102 nmap扫描 端口扫…

h5手写签名示例

前言 业务中需要用户进行签字&#xff0c;如何让用户在手机端进行签字&#xff1f; 示例如下 代码已分享至Gitee: https://gitee.com/lengcz/qianming 原示例&#xff1a; https://www.jq22.com/jquery-info13488 H5实现手写签字 创建一个html页面 <!DOCTYPE html> …

<Oracle>《Linux 下安装Oracle数据库 - Oracle 19C By CentOS 8 》(第一部分)

《Linux 下安装Oracle数据库 - Oracle 19C By CentOS 8 》&#xff08;第一部分&#xff09; 1 说明1.1 前言1.2 资源下载 2 安装步骤2.1 上传安装包2.2 下载数据库预安装包2.3 安装数据库预安装包 1 说明 1.1 前言 本文是Linux系统命令行模式安装Oracle数据库的学习实验记录…

【American English】实验室常用口语对话

不懂不丢人&#xff0c;不懂装懂才丢人。最近有点犯这毛病&#xff0c;多写一些东西消除一下。 无论什么知识都是多了才能成体系&#xff0c;更多自己在美国的小总结可见专栏&#xff1a;English。 文章目录 初次见面日常问候找寻某个东西找寻某个人 初次见面 Nice to meet you…

Linux之tar安装

目录 Linux之tar安装 定义 工作过程 语法格式 参数及用法 使用源代码安装软件的优点 注意&#xff1a;源代码编译环境 操作流程 解包 —— tar 配置 —— ./configure 编译 —— make 安装 —— make install 案例 --- 安装Apache服务 1.获取安装包地址并下载 2…

衣服面料相关基础

总结自 BiliBili视频&#xff1a;原来衣服的面料还能这么选&#xff0c;几个方法教你买到优质的短袖&#xff0c;再也不怕买衣服踩坑了 面子里子 既不能皱巴巴 又不能不透气 混纺 涤纶 粘纤 氨纶 涤纶 不变性 挺阔感 氨纶 弹性 粘纤 吸水透气40-50% 怕热 真丝与亚麻 …

Python 进阶(三):Python使用ORM框架SQLAlchemy操作Oracle数据库

Python使用ORM框架SQLAlchemy操作Oracle数据库 前言1. 安装Oracle Instant Client2. 安装依赖库3. 导入模块并创建引擎4. 操作oracle数据库4.1 新增数据4.2 查询数据4.3 更新数据4.4 删除数据 前言 要详细连接Oracle数据库并使用SQLAlchemy进行操作&#xff0c;按照以下步骤进…

云计算与OpenStack简介

文章目录 云计算与OpenStack简介什么是云服务模式部署模型 Openstac概述Openstack服务组件 云计算与OpenStack简介 什么是云 云是一种服务&#xff0c;就像我们去餐厅吃饭一样&#xff0c;只需要点菜&#xff0c;不需要知道厨师怎样烹饪食物。在云中&#xff0c;用户也只需要…

chatgpt赋能python:Python数据预处理:优化数据分析的重要步骤

Python数据预处理&#xff1a;优化数据分析的重要步骤 在数据分析过程中&#xff0c;数据预处理是非常重要的步骤。在这个阶段&#xff0c;我们可以清洗、转换和整理数据&#xff0c;以便更好地进行数据分析和建模。Python是一个强大的工具&#xff0c;可以帮助我们优化数据预…

Neural network-based clustering using pairwise constraints (ICLR-workshop 2016)

Neural network-based clustering using pairwise constraints (ICLR-workshop 2016) 源代码 摘要 这篇论文提出了一个基于神经网络的端到端的聚类框架。我们设计了一种新策略&#xff0c;除了学习适用于聚类的特征嵌入&#xff0c;还直接在源数据利用对比方法来推动数据形成…

计算机科学与技术报考指南【河南农业大学】

文章目录 前言环境介绍龙子湖校区文化路校区许昌校区 学院专业介绍计算机学习方面思维上态度上 最后 前言 这几年计算机分数虚高已经成为了不争的事实&#xff0c;加上计算机技术发展日新月异、大多数高校学习课程落后&#xff0c;转换思维另辟蹊径的报考农林类高校的计算机专…

WebSocket:基于 Spring Cloud 配置注解实现 WebSocket 集群方案

上一篇&#xff1a;WebSocket 的具体介绍与内部执行原理 文章目录 介绍用法抽象思路转发思路连接流程获取服务实例信息连接区分和管理 消息发送连接选择给指定用户发送消息给指定路径发送消息 结束源码地址声明 介绍 WebSocket大家应该是再熟悉不过了&#xff0c;如果是单体应…

Debezium系列之:发布Debezium 2.3.0.Final

Debezium系列之&#xff1a;发布Debezium 2.3.0.Final 一、重大变化1.PostgreSQL / MySQL 安全连接更改2.JDBC 存储编码更改 二、新功能和改进1.Debezium Server支持K8s2.新的通知子系统3.新的可扩展信号子系统4.JMX 信号和通知集成5.新的 JDBC 存储子系统6.PostgreSQL 流式传输…