Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >【Spring专场】「IOC容器」不看源码就带你认识核心流程以及运作原理

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

原创
作者头像
码界西柚
发布于 2022-01-08 04:37:36
发布于 2022-01-08 04:37:36
3440
举报

这是史上最全面的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别名。
  2. 第二步将存在根据 名称或者别名进行获取相关的缓存池找那个进行获取相关的对象实例
    • 如果存在: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。
  3. 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. 一般默认而言,会使用最少参数的构造器,当然如果存在默认构造器,一般会采用默认构造器区进行处理,但是如果存在非默认的构造器,则会采用参数注入的方式进行构造器进行构建。
  5. 核心: 我们前面已经将构造器列表进行排序完成后,会进行筛选获取合适的构造器进行执行构建对象。如果我们获取到了一个含有参数的构造器,那么spring框架会怎么做?
    • 先进行获取相关构造器中的所有相关的形式参数的名称以及类型。
    • 在进行解析参数,此解析方式会将对一些已经保存在容器中的数据进行解析注入以及相关的类型参数转换机制。
    • 从而计算构造器与数值类型的差异性,选择最佳何时的构造器方法。
    • 当我们已经筛选出和是的构造方法(最终),如果在此使用创建bean对象实例的时候,可以直接使用,无需在进行筛选。
    • 之后我们采用初始化策略进行构建该实例bean对象。
    • 最后将该对象注入到我们的BeanWrapperImpl对象模型中,并返回对象。
如果通过构造器或者工厂方法都无法构建

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

  1. 通过工厂方法进行构建
  2. 通过自定义构造器进行构建
  3. 通过默认构造器进行构建
构建的方式需要配合动态代理机制

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

解决循环依赖

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

IOC容器篇

主要的方法为populateBean方法

popluteBean的方法的执行流程

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

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

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

  1. 之后还会在采用后置处理器对属性进行动态pvs的内容进行填充处理。
  2. 会将属性应用到bean中的applyProperyValues方法:
代码语言:txt
AI代码解释
复制
- 在检测属性值是否已经完成转换,如果该属性值已经完成转换,则直接使用,无需再次转换。
- 遍历属性列表,解析器属性的原始值,在通过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方法。
  2. 之后开始执行初始化的前置操作:例如BeanPostProcessor以及相关的afterPropertiesSetting方法。
  3. 执行相关的初始化操作invokeInitMethods方法。
  4. 执行后置的初始化操作,例如BeanPostProcessor的后置处理机制操作。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Ioc容器refresh总结(5)--- Spring源码从入门到精通(三十五 )
先创建applicationEventMulticaster多播(派发)器,再注册registerListener添加到派发器中,之后下篇文章将介绍如何调用spring监听,这篇文章先介绍bean创建,因为监听在bean创建完成之后调用。
用户9919783
2022/07/26
2610
Ioc容器refresh总结(5)--- Spring源码从入门到精通(三十五 )
编织Spring魔法:解读核心容器中的Beans机制【beans 一】
在现代软件开发领域,Spring框架已经成为构建强大、灵活、可维护应用程序的不二选择。而Spring的核心容器是这一框架的心脏,Beans则是其中最为关键的组成部分。本文将带您深入探索这个令人着迷的世界,解锁Beans的神秘面纱,让您更深刻地理解Spring的精髓。
一只牛博
2025/05/30
930
编织Spring魔法:解读核心容器中的Beans机制【beans 一】
Spring源码系列:核心概念解析
本文旨在为读者解析Spring源码中的关键类,以便读者在深入阅读源码时,能够了解关键类的作用和用途。在阅读Spring源码时,经常会遇到一些不熟悉的概念,了解关键类的作用可以帮助读者更好地理解这些概念。
努力的小雨
2024/04/28
1210
【Spring专场】「MVC容器」不看源码就带你认识核心流程以及运作原理
之前已经写了很多问斩针对于SpringMVC的的执行原理和核心流程,在此再进行冗余介绍就没有任何意义了,所以我们主要考虑的就是针对于SpringMVC还没但大框架有介绍的相关内容解析分析和说明,那么接下来就让我们来接入Spring框架在核心的三大框架之一。
码界西柚
2022/01/10
3090
【Spring专场】「MVC容器」不看源码就带你认识核心流程以及运作原理
Spring源码核心知识点凝练总结
IOC全称 Inversion Of Control ,意为控制反转,通过IOC容器,我们把对象或者组件的创建过程透明化;我们无需关注创建细节,我们只需要把创建对象,属性赋值,这些工作交给IOC容器完成即可。
大忽悠爱学习
2023/03/14
6780
Spring源码核心知识点凝练总结
【Spring专场】「AOP容器」不看源码就带你认识核心流程以及运作原理
前一篇文章主要介绍了spring核心特性机制的IOC容器机制和核心运作原理,接下来我们去介绍另外一个较为核心的功能,那就是AOP容器机制,主要负责承接前一篇代理模式机制中动态代理:JDKProxy和CglibProxy的功能机制之后,我们开始研究一下如何实现一下相关的AOP容器代理机制的。
码界西柚
2022/01/09
5330
【Spring专场】「AOP容器」不看源码就带你认识核心流程以及运作原理
【Spring源码解读 底层原理高级进阶】【上】探寻Spring内部:BeanFactory和ApplicationContext实现原理讲解
本篇文章的目标,在于通过深入剖析Spring框架的源码,不仅让读者亲身体验到Spring设计大师们的卓越智慧,同时也分享我对其设计哲学所引发的思考与感悟。
苏泽
2024/03/01
3770
面试专题:Spring核心容器IOC的理解、原理与实现
本文主要讲解Spring核心容器之一IOC(Inversion of Control,控制反转)。IOC是一种设计模式,它是依赖注入(Dependency Injection,DI)的一种实现方式。在Spring框架中,IOC是一种用于实现依赖注入的技术,它可以帮助开发人员更好地管理和组织代码,提高代码的可重用性和可维护性。IOC也是在面试中经常问到的一个点,本文就来讲一下对IOC理解、原理与实现。
小明爱吃火锅
2023/12/22
3.6K0
Spring解决循环依赖
1、Husband创建Bean,先判断缓存池中是否存在,存在直接返回,不存在进入createBean创建的流程,调用构造方法创建一个早期的Bean【未进行属性赋值】,创建成功将其放入二级缓存earlySingletonObjects中,之后又调用addSingletonFactory方法将其放入三级缓存中并且将二级缓存中的移除,之后调用populateBean为属性赋值,在@Autowired的后置处理器中查找需要注入的依赖,发现Husband中的一个属性Wife,因此调用getBean方法从容器中获取,但是此时的Wife还未创建,因此又进入了doGetBean的流程,但是此时Wife并没有创建,因此在一二三级缓存中不能获取,又执行createBean方法创建Wife,同样调用构造方法创建一个早期Bean放入二级缓存中,调用addSingletonFactory放入三级缓存并移除二级缓存,然后调用populateBean方法为Wife属性赋值,在@Autowired的后置处理器中查找需要注入的依赖,发现Wife类中有一个属性是Husband,因此调用getBean方法,再次调用doGetBean获取Husband,但是此时的Husband已经创建成功【未赋值】,存放在三级缓存中,因此直接从三级缓存中取出Husband赋值给Wife属性,至此Wife属性已经赋值成功,直接添加到一级缓存(singletonObjects)中并且移除三级缓存,直接返回给Husband赋值,因此Husband中的属性也持有了Wife的引用,都创建并且赋值成功了。
爱撒谎的男孩
2019/12/31
9520
Spring-IoC容器工作原理
容器底层用DefaultListableBeanFactory,即实现了BeanDe finitionRegistry,又实现了BeanFactory java配置 :
茶半香初
2021/11/26
2990
Spring-IoC容器工作原理
Spring 框架核心原理
尽管希腊哲学家赫拉克利特(Heraclitus)并不作为一名软件开发人员而闻名,但他似乎深谙此道。他的一句话经常被引用:“唯一不变的就是变化”,这句话抓住了软件开发的真谛。
一个会写诗的程序员
2020/03/20
1.1K0
Spring 框架核心原理
Spring MVC系列-(7) IOC初始化流程
IoC容器的初始化就是含有BeanDefinition信息的Resource的定位、载入、解析、注册四个过程,最终我们配置的bean,以beanDefinition的数据结构存在于IoC容器即内存中。这里并不涉及bean的依赖注入,只是bean定义的载入。但有例外,在使用Ioc容器时有一个预实例化的配置,即bean定义中的设置了lazyinit属性,那么这个bean在Ioc容器初始化时就预先加载,不需要等到Ioc整个初始化后,第一次getBean时才会触发。其中refresh()启动对Ioc容器的初始化。
码老思
2023/10/19
5850
Spring MVC系列-(7) IOC初始化流程
这次彻底搞懂IoC容器依赖注入的源码
初始化的过程,主要完成的工作是在容器中建立 BeanDefinition 数据映射,并没有看到容器对Bean依赖关系进行注入。
JavaEdge
2021/10/18
4190
Spring源码解析(四):单例bean的创建流程
Java微观世界
2025/01/21
2410
Spring源码解析(四):单例bean的创建流程
SpringBoot源码解析(八):Bean工厂接口体系
在前文中,我们详细概述了SpringBoot架构中应用上下文结构体系的各个组件。其中,通用应用上下文(GenericApplicationContext)通过创建 DefaultListableBeanFactory,负责注册、管理和创建Bean定义,同时支持依赖注入和动态Bean管理。DefaultListableBeanFactory结构相对复杂,接下来我们将对其组件逐一进行介绍。
Java微观世界
2025/02/02
1470
SpringBoot源码解析(八):Bean工厂接口体系
Spring IOC 容器源码分析 - 创建单例 bean 的过程
在上一篇文章中,我比较详细的分析了获取 bean 的方法,也就是getBean(String)的实现逻辑。对于已实例化好的单例 bean,getBean(String) 方法并不会再一次去创建,而是从缓存中获取。如果某个 bean 还未实例化,这个时候就无法命中缓存。此时,就要根据 bean 的配置信息去创建这个 bean 了。相较于getBean(String)方法的实现逻辑,创建 bean 的方法createBean(String, RootBeanDefinition, Object[])及其所调用的方法逻辑上更为复杂一些。关于创建 bean 实例的过程,我将会分几篇文章进行分析。本篇文章会先从大体上分析 createBean(String, RootBeanDefinition, Object[])方法的代码逻辑,至于其所调用的方法将会在随后的文章中进行分析。
田小波
2018/06/04
7640
Spring IOC 容器源码分析 - 创建单例 bean 的过程
Spring的Bean加载流程
Spring IOC容器就像是一个生产产品的流水线上的机器,Spring创建出来的Bean就好像是流水线的终点生产出来的一个个精美绝伦的产品。既然是机器,总要先启动,Spring也不例外。因此Bean的加载流程总体上来说可以分为两个阶段:
全栈程序员站长
2022/06/29
5080
Spring的Bean加载流程
Spring IOC知识点一网打尽!
前言 回顾前面: 给女朋友讲解什么是代理模式 包装模式就是这么简单啦 单例模式你会几种写法? 工厂模式理解了没有? 在刷Spring书籍的时候花了点时间去学习了单例模式和工厂模式,总的来说还是非常值得的! 本来想的是刷完《Spring 实战 (第4版)》和《精通Spring4.x 企业应用开发实战》的IOC章节后来重新编写一篇IOC的文章的,看了一下之前已经写过的入门系列Spring入门这一篇就够了和Spring【依赖注入】就是这么简单。最主要的知识点都已经讲过了,所以感觉就没必要重新来编写这些知识点了…
Java3y
2018/06/11
6320
【小家Spring】为脱离Spring IOC容器管理的Bean赋能【依赖注入】的能力,并分析原理(借助AutowireCapableBeanFactory赋能)
可能我们(大众)都是这样认为:自从用上了Spring这个优秀的框架,一般小伙伴们都是一言不合就把对象塞进Spring的IOC容器里面,交给它来替我们管理。
YourBatman
2019/09/03
3.2K0
【小家Spring】为脱离Spring IOC容器管理的Bean赋能【依赖注入】的能力,并分析原理(借助AutowireCapableBeanFactory赋能)
Spring IOC源码解读
Spring IOC源码解读 此文讲解非常详细,内容较多,请耐心看完,谢谢 讲解内容: 一、 什么是Ioc/DI? 二、 Spring IOC体系结构 (1) BeanFactory (2) BeanDefinition 三、 IoC容器的初始化 1、 XmlBeanFactory(屌丝IOC)的整个流程 2、 FileSystemXmlApplicationContext 的IOC容器流程 2.1、高富帅IOC解剖 2.2、 设置资源加载器和资源定位 2.3、AbstractApplicationCont
Java帮帮
2018/03/15
1.5K0
Spring IOC源码解读
推荐阅读
相关推荐
Ioc容器refresh总结(5)--- Spring源码从入门到精通(三十五 )
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档