0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

解析加载及实例化Bean的顺序(零配置)

5jek_harmonyos 来源:CSDN 作者:低调的JVM 2021-08-04 16:08 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

作者丨低调的JVM

来自丨CSDN

https://blog.csdn.net/qq_27529917/article/details/79329809

在使用Spring时,Bean之间会有些依赖,比如一个Bean A实例化时需要用到Bean B,那么B应该在A之前实例化好。很多时候Spring智能地为我们做好了这些工作,但某些情况下可能不是,比如Springboot的@AutoConfigureAfter注解,手动的指定Bean的实例化顺序。

了解Spring内Bean的解析,加载和实例化顺序机制有助于我们更好的使用Spring/Springboot,避免手动的去干预Bean的加载过程,搭建更优雅的框架。

Spring容器在实例化时会加载容器内所有非延迟加载的单例类型Bean,看如下源码:

public abstract class AbstractApplicationContext extends DefaultResourceLoader

implements ConfigurableApplicationContext, DisposableBean {

//刷新Spring容器,相当于初始化

public void refresh() throws BeansException, IllegalStateException {

。。。。。。

// Instantiate all remaining (non-lazy-init) singletons.

finishBeanFactoryInitialization(beanFactory);

}

}

public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory

implements ConfigurableListableBeanFactory, BeanDefinitionRegistry, Serializable {

/** List of bean definition names, in registration order */

private volatile List《String》 beanDefinitionNames = new ArrayList《String》(256);

public void preInstantiateSingletons() throws BeansException {

List《String》 beanNames = new ArrayList《String》(this.beanDefinitionNames);

for (String beanName : beanNames) {

。。。。。。

getBean(beanName); //实例化Bean

}

}

}

ApplicationContext内置一个BeanFactory对象,作为实际的Bean工厂,和Bean相关业务都交给BeanFactory去处理。

在BeanFactory实例化所有非延迟加载的单例Bean时,遍历beanDefinitionNames 集合,按顺序实例化指定名称的Bean。beanDefinitionNames 属性是Spring在加载Bean Class生成的BeanDefinition时,为这些Bean预先定义好的名称,看如下代码:

public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory

implements ConfigurableListableBeanFactory, BeanDefinitionRegistry, Serializable {

public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)

throws BeanDefinitionStoreException {

。。。。。。

this.beanDefinitionNames.add(beanName);

}

}

BeanFactory在加载一个BeanDefinition(也就是加载Bean Class)时,将相应的beanName存入beanDefinitionNames属性中,在加载完所有的BeanDefinition后,执行Bean实例化工作,此时会依据beanDefinitionNames的顺序来有序实例化Bean,也就是说Spring容器内Bean的加载和实例化是有顺序的,而且近似一致,当然仅是近似。

Spring在初始化容器时,会先解析和加载所有的Bean Class,如果符合要求则通过Class生成BeanDefinition,存入BeanFactory中,在加载完所有Bean Class后,开始有序的通过BeanDefinition实例化Bean。

我们先看加载Bean Class过程,零配置下Spring Bean的加载起始于ConfigurationClassPostProcessor的postProcessBeanDefinitionRegistry(BeanDefinitionRegistry)方法,我总结了下其加载解析Bean Class的流程:

配置类可以是Spring容器的起始配置类,也可以是通过@ComponentScan扫描得到的类,也可以是通过@Import引入的类。如果这个类上含有@Configuration,@Component,@ComponentScan,@Import,@ImportResource注解中的一个,或者内部含有@Bean标识的方法,那么这个类就是一个配置类,Spring就会按照一定流程去解析这个类上的信息。

在解析的第一步会校验当前类是否已经被解析过了,如果是,那么需要按照一定的规则处理(@ComponentScan得到的Bean能覆盖@Import得到的Bean,@Bean定义的优先级最高)。

如果未解析过,那么开始解析:

解析内部类,查看内部类是否应该被定义成一个Bean,如果是,递归解析。

解析@PropertySource,也就是解析被引入的Properties文件。

解析配置类上是否有@ComponentScan注解,如果有则执行扫描动作,通过扫描得到的Bean Class会被立即解析成BeanDefinition,添加进beanDefinitionNames属性中。之后查看扫描到的Bean Class是否是一个配置类(大部分情况是,因为标识@Component注解),如果是则递归解析这个Bean Class。

解析@Import引入的类,如果这个类是一个配置类,则递归解析。

解析@Bean标识的方法,此种形式定义的Bean Class不会被递归解析

解析父类上的@ComponentScan,@Import,@Bean,父类不会被再次实例化,因为其子类能够做父类的工作,不需要额外的Bean了。

在1,3,4,6中都有递归操作,也就是在解析一个Bean Class A时,发现其上能够获取到其他Bean Class B信息,此时会递归的解析Bean Class B,在解析完Bean Class B后再接着解析Bean Class A,可能在解析B时能够获取到C,那么也会先解析C再解析B,就这样不断的递归解析。

在第3步中,通过@ComponentScan扫描直接得到的Bean Class会被立即加载入beanDefinitionNames中,但@Import和@Bean形式定义的Bean Class则不会,也就是说正常情况下面@ComponentScan直接得到的Bean其实例化时机比其他两种形式的要早。

通过@Bean和@Import形式定义的Bean Class不会立即加载,他们会被放入一个ConfigurationClass类中,然后按照解析的顺序有序排列,就是图片上的 “将配置类有序排列”。一个ConfigurationClass代表一个配置类,这个类可能是被@ComponentScan扫描到的,则此类已经被加载过了;也可能是被@Import引入的,则此类还未被加载;此类中可能含有@Bean标识的方法。

Spring在解析完了所有Bean Class后,开始加载ConfigurationClass。如果这个ConfigurationClass是被Import的,也就是说在加载@ComponentScan时其未被加载,那么此时加载ConfigurationClass代表的Bean Class。然后加载ConfigurationClass内的@Bean方法。

顺序总结:@ComponentScan 》 @Import 》 @Bean

Bean Class的结构图如上所示,A是配置类的入口,通过A能直接或间接的引入一个模块。

此时启动Spring容器,将A引入容器内。

如果A是通过@ComponentScan扫描到的,那么此时的加载顺序是:

A 》 D 》 F 》 B 》 E 》 G 》 C

如果A是通过@Import形式引入的,那么此时的加载顺讯是:

D 》 F 》 B 》 E 》 G 》 A 》 C

当然以上仅仅代表着加载Bean Class的顺序,实际实例化Bean的顺序和加载顺序大体相同,但还是会有一些差别。

Spring在通过getBean(beanName)形式实例化Bean时,会通过BeanDefinition去生成Bean对象。在这个过程中,如果BeanDefinition的DependsOn不为空,从字面理解就是依赖某个什么,其值一般是某个或多个beanName,也就是说依赖于其他Bean。

此时Spring会将DependsOn指定的这些名称的Bean先实例化,也就是先调用getBean(dependsOn)方法。我们可以通过在Bean Class或者@Bean的方法上标识**@DependsOn**注解,来指定当前Bean实例化时需要触发哪些Bean的提前实例化。

当一个Bean A内部通过@Autowired或者@Resource注入Bean B,那么在实例化A时会触发B的提前实例化,此时会注册A》B的dependsOn依赖关系,实质和@DependsOn一样,这个是Spring自动为我们处理好的。

了解Spring Bean的解析,加载及实例化的顺序机制能够加深对Spring的理解,搭建更优雅简介的Spring框架。

编辑:jq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • spring
    +关注

    关注

    0

    文章

    345

    浏览量

    16134

原文标题:Spring解析,加载及实例化Bean的顺序(零配置)

文章出处:【微信号:harmonyos_developer,微信公众号:harmonyos_developer】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    VirtualLab Fusion中的可视设置

    选项可以通过以下控件进行重置、加载和保存: **主窗口设置 ** **字体配置 ** **数字显示 ** **文档窗口设置 ** **1D数据数组可视设置 ** **颜色表 ** **谐波场视图 ** **文件信息 ** 延
    发表于 06-03 08:28

    EB Tresos 中缺少加密模块是否为启用了 HSE 的 S32G3 引导加载程序提供参考加密配置

    恩智浦团队大家好, 我正在做S32G3 引导加载程序 (bootloader_s32g3xx_asr_4.4_m7)用EB Tresos 工作室与加密模块(HSE 集成)。 在配置
    发表于 05-22 07:11

    i.MX93定制板 – DDR 配置与引导加载程序刷新顺序的疑问求解

    的,还是应该先刷新引导加载程序? - 如果 DDR 配置确实是第一步,为什么配置工具验证在我们的自定义板上会失败,而在 EVK 上工作正常? 有关此 USB 通信问题的正确启动顺序和潜
    发表于 04-16 06:27

    IDT72T6480:2.5V 顺序流控制设备的深度解析

    IDT72T6480:2.5V 顺序流控制设备的深度解析 在电子设计领域,高效的数据存储与处理一直是工程师们追求的目标。IDT72T6480 这款 2.5V 顺序流控制设备,凭借其独特的特性和广泛
    的头像 发表于 04-12 12:40 637次阅读

    ELF-RV112B RKNN模型加载与运行时初始

    ELF-RV112B RKNN模型加载与运行时初始
    的头像 发表于 04-03 16:08 300次阅读
    ELF-RV112B RKNN模型<b class='flag-5'>加载</b>与运行时初始<b class='flag-5'>化</b>

    碳园区数字平台赋能型模式的适用场景解析

    碳园区数字平台赋能型模式,核心是依托能碳一体数字平台,打通“源-网-荷-储-碳”全链路数据壁垒,通过智能感知、建模分析、动态调控与协同管理,为园区碳转型提供精准
    的头像 发表于 03-27 11:44 309次阅读
    <b class='flag-5'>零</b>碳园区数字<b class='flag-5'>化</b>平台赋能型模式的适用场景<b class='flag-5'>解析</b>

    电磁兼容仿真模拟系统平台:全景解析与应用实例

    电磁兼容仿真模拟系统平台:全景解析与应用实例
    的头像 发表于 03-10 10:39 630次阅读
    电磁兼容仿真模拟系统平台:全景<b class='flag-5'>解析</b>与应用<b class='flag-5'>实例</b>

    掌握 LuatIO:GPIO 复用模式初始配置全流程解析

    在使用 LuatIO 进行嵌入式应用开发时,合理配置 GPIO 的复用功能是实现外设控制的前提。本文全面解析 GPIO 引脚由普通 IO 转换为复用功能引脚的初始流程,包括时钟使能、模式选择、速度
    的头像 发表于 01-23 15:28 3065次阅读
    掌握 LuatIO:GPIO 复用模式初始<b class='flag-5'>化</b><b class='flag-5'>配置</b>全流程<b class='flag-5'>解析</b>

    奥拓电子数字售解决方案实现全球拓展与规模应用

    随着人工智能与数字技术深度融入售场景,线下售空间正经历从“数字展示”到“智能运营”的关键演进。奥拓电子锚定“AI+智能视讯”核心战略,打造全链路新
    的头像 发表于 01-20 11:00 933次阅读

    PCBA加工件封装技术解析:从传统到前沿的全面指南

    的性能、可靠性和小型程度。以下从封装类型、技术特点、应用场景及发展趋势四个方面进行系统解析。   PCBA加工件封装技术解析 一、主流封装技术类型 PCBA
    的头像 发表于 12-26 09:46 719次阅读

    图扑软件 3D 场景预加载应用实现

    加载是在进入正式场景之前提前加载所需模型、材质、图片等资源的技术手段,其核心价值在于消除资源加载等待,确保场景首次渲染即可完整呈现,从而提供无缝、流畅的用户体验。在复杂的 Web 3D 可视
    的头像 发表于 12-01 16:04 1368次阅读
    图扑软件 3D 场景预<b class='flag-5'>加载</b>应用实现

    Linux内核模块的加载机制

    ) __versions 内核符号版本校验数据 __ksymtab导出的符号表 .init.text 初始函数(模块加载时执行) .exit.text 清理函数(模块卸载时执行)首先解析ELF头: 1、提取
    发表于 11-25 06:59

    软通动力携手华为云推动制造与售行业数字转型

    近日,软通动力携手华为云在上海共同举办“智链未来、数创增长——制造与售行业AI应用及数据治理解决方案研讨会”。活动聚焦AI大模型构建与数据治理两大核心议题,深入解析企业智能应用落地的关键路径,为制造与售企业探索数字
    的头像 发表于 10-27 17:29 1629次阅读

    广凌智慧教室场景应用:适配多元教学需求的定制配置解析

    在教育信息2.0时代,智慧教室的建设已从单一功能升级转向场景、个性、生态的深度融合。广凌科技基于多年技术积累与实践验证,提出覆盖“常态
    的头像 发表于 07-18 10:00 743次阅读
    广凌智慧教室场景应用:适配多元教学需求的定制<b class='flag-5'>化</b><b class='flag-5'>配置</b><b class='flag-5'>解析</b>

    鸿蒙5开发宝藏案例分享---Web加载时延优化解析

    : Network泳道 :查看资源加载时序 Main泳道 :监控JS/CSS解析阻塞 Performance面板 :定位长任务(Long Tasks) ?️** 四大优化方向 + 代码实战** 以下
    发表于 06-12 17:11