【Spring 专场】「IOC 容器」不看源码就带你认识核心流程以及运作原理

本文涉及的产品
容器镜像服务 ACR,镜像仓库100个 不限时长
简介: 【Spring 专场】「IOC 容器」不看源码就带你认识核心流程以及运作原理

这是史上最全面的Spring的核心流程以及运作原理的分析指南


  • 🍃【Spring核心专题】「IOC容器篇」不看繁琐的源码就带你浏览Spring的核心流程以及运作原理


  • 🍃【Spring核心专题】「AOP容器篇」不看繁琐的源码就带你浏览Spring的核心流程以及运作原理


  • 🍃【Spring核心专题】「MVC容器篇」不看繁琐的源码就带你浏览Spring的核心流程以及运作原理




学好Spring技术的背景


针对于每一个Java的爱好者而言,无论是从事面向于微服务架构技术的领域(SpringCloud、SpringCloud-Alibaba等),还是面向于传统互联网行业(SpringBoot)以及软件系统


(Spring\SpringBatch)领域,掌握好Spring框架技术原理和源码对排查问题以及未来的面试技术有着非常重要的帮助和影响,而接下来,笔者会针对于Spring的技术框架的核心源码流程点进行相关的分析和认识,相信阅读完本篇文章,一定会对Spring的源码和执行原理有着很大的帮助和提升。




分析框架核心流程


获取Spring框架的IOC容器


IOC容器执行流程主要核心流程点:


  • 获取单例Bean对象
  • 创建单例Bean对象
  • 创建原始Bean对象
  • 解决循环依赖
  • 填充属性信息
  • 初始化Bean对象


getBean方法的执行流程


  1. 第一步将beanName或者BeanType类型进行获取相关的容器数据对象,例如:处理以&符号开头的name名称数据,以及根据相关的alias别名。


  1. 第二步将存在根据 名称或者别名进行获取相关的缓存池找那个进行获取相关的对象实例


  • 如果存在:Spring框架会调用getObjectForBeanInstance方法,返回对应的Bean实例对象,其中Bean实例的类型有两种模式:单例模式和原型模式
  • 单例模式:缓存中没有,创建一个,然后放入缓存中,其中会对该单例对象bean进行先关的拦截和后置工作。
  • 原型模式:每次都会创建新的对象进行返回相关的对象。
  • 如果当前的容器中,无法获取到相关的对应的BeanName的对象实例,则会进行想父容器进行寻找对应的对象Bean实例,如果父容器中存在,直接返回父容器中的数据对象实例,但是如果父容器还不存在,则会进行创建Bean对象实例了,但是在创建之前,会进行解析两种特殊的Bean操作关系。



两种特殊的Bean实例的关联关系


  • parent bean的继承关系,例如,a bean对象可以在xml文件中继承相关 a-parent bean的属性以及相关的覆盖操作
  • 处理相关的depend-ons依赖关系操作,这样子可以根据依赖关系,建立一个加载和创建Bean之间的前后关系和依赖关系,例如A depend-ons B的bean对象,那么在创建A之前一定会先加载和创建B,依此类托。



之后进行相关的创建bean的操作控制!



获取Spring框架的变量容器


  • singletonObjects:单例一级缓存池-用于存放完全实例化+初始化好的对象Bean,如果从该缓存池中取出的Bean可以直接的使用。
  • earlySingletonObject:单例二级缓存池-用于存放正在初始化的对象bean,主要用于解决循环依赖的临时存放的对象池。
  • singletonFactories:用于存放bean对象的工厂对象机制,主要用于创建bean对象的ObjectFactory。



createBean方法的执行流程


  • createBean的方法入口,getSIngleton方法:


  1. 先从singletonObjects集合中获取相关的Bean实例,若不为空,则直接返回。
  2. 如果获取不到相关的对象实例在一级单例缓存池中,则会进行createBeanInstance实例阶段(此部分,接下来会详细介绍),会将对应的BeanName添加到singleCurrentlyInCreation集合中,这个集合主要用于存放相关的将要创建的对象bean,这个是第一步。
  3. 当通过getObject方法调用createBean方法的是创建实例对象的完成之后,会将对象实例从singleCurrentlyInCreation集合中进行转移到singleObjects对象集合缓存池中,映射关系为:beanName->singleObject对象。



createBean的方法要点


解析Bean的类型和属性类型特点分析,主要分为以下几点内容:


  1. 解析相关的Bean对象的类型。
  2. 校验和分析处理相关的override注解修饰的方法,主要用于先去校验和分析是否存在重载方法或者覆盖方法,方便cglib动态代理的时候不需要进行校验,而是直接处理调用即可。
  • 其中有一个属性:lookup-method,如果我们希望在单例对象里面加入一个原型模式(prototype)的对象属性,那么可以考虑使用<lookup-method name="getPrototypeBean", bean = "prototypeBean" /> ApplicationContextAware。
  1. bean实例化前的后置处理控制hook钩子函数以及相关回调机制控制。



createBean的最核心方法doCreateBean


  • 调用doCreateBean创建bean实例,此方法算是最底层的创建createBean的代表方法了,首先他会遵循从缓存中区获取相关的BeanWrapper实现类对象,并且清除一些临时数据信息。
  • 如果缓存中没有相关的缓存,则会进行手动创建bean实例对象,将实例对象包裹在BeanWrapper实例类对象并且返回该BeanWrapper对象。
  • 并且采用MergeBeanDefinitionBeanPostProcessor的后置处理器,对相关的对象的abstract和parent的继承关系的bean进行合并处理。
  • 根据系统的配置是否支持循环依赖的选项,进行选择和决定是否采用提前暴露bean的早期引用(early reference),主要用于处理的循环依赖。
  • 之后对相关的提前暴露的引用和属性字段进行使用popluateBean方法进行引用的属性进行填充,其中也包含了相关的循环引用的概念在里面。
  • 调用相关的initializeBean方法完成余下的初始化工作任务,包含了:initializeBean接口实现、@PostConstruct注解处理控制、以及init-method方法的属性处理。
  • 注册销毁相关的distroy-method的属性以及相关的preDestory的方法控制。



doCreateBean创建最原始的Bean对象


主要通过createBeanInstance方法实例机制,其核心流程为:


  • 检测类的访问权限,若禁止访问,则会抛出异常机制。
  • 如果该对象bean的factory-method属性包含了factory工厂方法机制不为空,则通过该定义的声明的相关的factory方法进行创建bean,并且返回结果。



通过相关的构造器的方式进行构建对象


在此我们会采用construct的方式进行反射进行构建实例对象,并且返回对象的对象结果,步骤如下:

  1. 创建相关的BeanWrapperImpl对象作为先关的Bean实例对象的包装实现类。

  2. 之后需要进行构建相关的真实的原始模型对象,其中上面说了,如果该bean定义拥有相关的factory方法,则会直接通过factory方法建立,否则会采用构造器的方式进行构建哦!
  3. 会针对于该对象的所有定义以及隐含的构造器进行分析和处理,采用minOrArg方式计算出,进行分析出了一个按照参数数量进行排序的构造器列表。(其中会包含着访问优先级以及参数个数的条件进行排序)。
  4. 一般默认而言,会使用最少参数的构造器,当然如果存在默认构造器,一般会采用默认构造器区进行处理,但是如果存在非默认的构造器,则会采用参数注入的方式进行构造器进行构建。


  1. 核心: 我们前面已经将构造器列表进行排序完成后,会进行筛选获取合适的构造器进行执行构建对象。如果我们获取到了一个含有参数的构造器,那么spring框架会怎么做?


  • 先进行获取相关构造器中的所有相关的形式参数的名称以及类型。
  • 在进行解析参数,此解析方式会将对一些已经保存在容器中的数据进行解析注入以及相关的类型参数转换机制。
  • 从而计算构造器与数值类型的差异性,选择最佳何时的构造器方法。
  • 当我们已经筛选出和是的构造方法(最终),如果在此使用创建bean对象实例的时候,可以直接使用,无需在进行筛选。
  • 之后我们采用初始化策略进行构建该实例bean对象。
  • 最后将该对象注入到我们的BeanWrapperImpl对象模型中,并返回对象。



如果通过构造器或者工厂方法都无法构建


那么会采用组合方式进行构建该对象


  1. 通过工厂方法进行构建
  2. 通过自定义构造器进行构建
  3. 通过默认构造器进行构建



构建的方式需要配合动态代理机制


为了方便我们进行在对Springbean容器的对象进行AOP拦截操作处理机制。

image.png


解决循环依赖


话不多说,就是提前暴露,可以通过factory避过去以及@lazy不会引起错误等。



IOC容器篇


主要的方法为populateBean方法


popluteBean的方法的执行流程


首先会获取相关的注入该类对象bean的属性列表,我们再切定义为pvs。


  1. 当构造器构建完对象之后会进行相关的自定义属性进行填充,但是在进行相关的属性填充进行之前,会先去尝试采用系统默认后置处理器进行填充。


主要通过参数名或者参数类型进行解析并且填充相关的依赖属性,主要可以通过的手段就是@Autowired或者@Resource、@Inject等。


  1. 之后还会在采用后置处理器对属性进行动态pvs的内容进行填充处理。


  1. 会将属性应用到bean中的applyProperyValues方法:


  • 在检测属性值是否已经完成转换,如果该属性值已经完成转换,则直接使用,无需再次转换。
  • 遍历属性列表,解析器属性的原始值,在通过PropertisSourcePlaceholdConfigurer进行相关的解析操作,并且完成解析值resolveValue。
  1. 最后将的到的解析数值resolveValue进行相关的类型属性转换操作。
  2. 将类型转换后的值设置到PropertyValue对象中,将PropertyValue对象存入deepCopy集合中,并且将deepCopy的属性值注入到bean对象中。



根据名称和类型进行填充


根据名称注入


就是单纯的将bean名称进行注入到相关的非简单类型的注入机制。


根据类型注入


  • 主要处理@Value注解进行注入操作解析机制!
  • 解析数组、list、map等类型的依赖注入机制
  • 根据类型查找相关何时的类型数据信息
  • 如果候选项的数量为0,则抛出异常。如果=1,则直接从候选列表中进行获取,如果>1,则在多个候选选项中的获取最优的对象,否则抛出异常。
  • 如果候选选选为class类型,则标识候选选选还没有完成实例化,此时通过BeanFactory.getBean的方式进行实例化,否则会直接返回对象实例。



初始化Bean对象


主要是经历了所有的实例化和处理之后,则会需要进行相关的初始化方法的调用,在底层框架表现为initializeBean方法进行初始化,执行顺序的判断逻辑执行流程为:


  1. 检测bean是否实现了xAware类型的接口,如果实现了,则会向该bean中注入相关的x的实例属性对象,主要通过调用invokeAwareMethods方法。


  1. 之后开始执行初始化的前置操作:例如BeanPostProcessor以及相关的afterPropertiesSetting方法。


  1. 执行相关的初始化操作invokeInitMethods方法。


  1. 执行后置的初始化操作,例如BeanPostProcessor的后置处理机制操作。




相关文章
|
19天前
|
设计模式 Java 开发者
如何快速上手【Spring AOP】?从动态代理到源码剖析(下篇)
Spring AOP的实现本质上依赖于代理模式这一经典设计模式。代理模式通过引入代理对象作为目标对象的中间层,实现了对目标对象访问的控制与增强,其核心价值在于解耦核心业务逻辑与横切关注点。在框架设计中,这种模式广泛用于实现功能扩展(如远程调用、延迟加载)、行为拦截(如权限校验、异常处理)等场景,为系统提供了更高的灵活性和可维护性。
|
5月前
|
前端开发 Java 物联网
智慧班牌源码,采用Java + Spring Boot后端框架,搭配Vue2前端技术,支持SaaS云部署
智慧班牌系统是一款基于信息化与物联网技术的校园管理工具,集成电子屏显示、人脸识别及数据交互功能,实现班级信息展示、智能考勤与家校互通。系统采用Java + Spring Boot后端框架,搭配Vue2前端技术,支持SaaS云部署与私有化定制。核心功能涵盖信息发布、考勤管理、教务处理及数据分析,助力校园文化建设与教学优化。其综合性和可扩展性有效打破数据孤岛,提升交互体验并降低管理成本,适用于日常教学、考试管理和应急场景,为智慧校园建设提供全面解决方案。
368 70
|
4月前
|
XML Java 数据格式
Spring IoC容器的设计与实现
Spring 是一个功能强大且模块化的 Java 开发框架,其核心架构围绕 IoC 容器、AOP、数据访问与集成、Web 层支持等展开。其中,`BeanFactory` 和 `ApplicationContext` 是 Spring 容器的核心组件,分别定位为基础容器和高级容器,前者提供轻量级的 Bean 管理,后者扩展了事件发布、国际化等功能。
|
2月前
|
Java Spring 容器
SpringBoot自动配置的原理是什么?
Spring Boot自动配置核心在于@EnableAutoConfiguration注解,它通过@Import导入配置选择器,加载META-INF/spring.factories中定义的自动配置类。这些类根据@Conditional系列注解判断是否生效。但Spring Boot 3.0后已弃用spring.factories,改用新格式的.imports文件进行配置。
703 0
|
6月前
|
前端开发 Java 数据库
微服务——SpringBoot使用归纳——Spring Boot集成Thymeleaf模板引擎——Thymeleaf 介绍
本课介绍Spring Boot集成Thymeleaf模板引擎。Thymeleaf是一款现代服务器端Java模板引擎,支持Web和独立环境,可实现自然模板开发,便于团队协作。与传统JSP不同,Thymeleaf模板可以直接在浏览器中打开,方便前端人员查看静态原型。通过在HTML标签中添加扩展属性(如`th:text`),Thymeleaf能够在服务运行时动态替换内容,展示数据库中的数据,同时兼容静态页面展示,为开发带来灵活性和便利性。
285 0
|
2月前
|
缓存 JSON 前端开发
第07课:Spring Boot集成Thymeleaf模板引擎
第07课:Spring Boot集成Thymeleaf模板引擎
339 0
第07课:Spring Boot集成Thymeleaf模板引擎
|
6月前
|
XML Java 数据库连接
微服务——SpringBoot使用归纳——Spring Boot集成MyBatis——基于 xml 的整合
本教程介绍了基于XML的MyBatis整合方式。首先在`application.yml`中配置XML路径,如`classpath:mapper/*.xml`,然后创建`UserMapper.xml`文件定义SQL映射,包括`resultMap`和查询语句。通过设置`namespace`关联Mapper接口,实现如`getUserByName`的方法。Controller层调用Service完成测试,访问`/getUserByName/{name}`即可返回用户信息。为简化Mapper扫描,推荐在Spring Boot启动类用`@MapperScan`注解指定包路径避免逐个添加`@Mapper`
238 0
|
6月前
|
Java 测试技术 微服务
微服务——SpringBoot使用归纳——Spring Boot中的项目属性配置——少量配置信息的情形
本课主要讲解Spring Boot项目中的属性配置方法。在实际开发中,测试与生产环境的配置往往不同,因此不应将配置信息硬编码在代码中,而应使用配置文件管理,如`application.yml`。例如,在微服务架构下,可通过配置文件设置调用其他服务的地址(如订单服务端口8002),并利用`@Value`注解在代码中读取这些配置值。这种方式使项目更灵活,便于后续修改和维护。
86 0
|
6月前
|
SQL Java 数据库连接
微服务——SpringBoot使用归纳——Spring Boot使用slf4j进行日志记录—— application.yml 中对日志的配置
在 Spring Boot 项目中,`application.yml` 文件用于配置日志。通过 `logging.config` 指定日志配置文件(如 `logback.xml`),实现日志详细设置。`logging.level` 可定义包的日志输出级别,例如将 `com.itcodai.course03.dao` 包设为 `trace` 级别,便于开发时查看 SQL 操作。日志级别从高到低为 ERROR、WARN、INFO、DEBUG,生产环境建议调整为较高级别以减少日志量。本课程采用 yml 格式,因其层次清晰,但需注意格式要求。
546 0