Spring Bean创建和循环依赖
作者:? 发布时间:2023-10-21 09:17:51
1 前言
前文已经讲述了Spring BeanFactory 与 FactoryBean 的区别详情,在本文中将继续讲解 Bean 的创建和初始化,在这个环节中将会涉及到 Bean 的创建、初始化和循环依赖内容。
2 Bean 的创建
在前文中已经讲述了 Spring 容器启动的核心方法 refresh,关于 Bean 的创建和初始化方法都是在 finishBeanFactoryInitialization()
中进行处理,在这个阶段就是处理所有剩余的非懒加载的单例对象。
在该方法中,调用 preInstantiateSingletons()
进行 Bean 的所有单实例 bean。 在这个过程中,会获取容器中的所有 Bean, 依次进行初始化和创建对象。获取所有的 Bean 定义信息 beanDefinitionNames
。在处理 Bean 时需要判断 Bean 定义信息是不是抽象的,单例,和懒加载。其核心方法为 getBean , 也许大家都知道在获取 Bean 的过程中,会经历 getBean -> doGetBean -> createBean -> doCreateBean
方法调用链,在 Spring 源码中, doXXX 的方法都是实际业务的方法,在 doCreateBean
方法中,createBeanInstance
方法是真实创建 Bean 对象的方法,在 Spring 中,都是采用反射的方法来创建对象的。这些核心的方法都是在 AbstractAutowireCapableBeanFactory
中实现,下图便是 doCreateBean 方法,其中的核心操作有三个: createBeanInstance 、populateBean、initializeBean
。
createBeanInstance
createBeanInstance
是创建 Bean 对象的方法,这里最终调用的是 instantiateBean
方法,最终的调用栈如下:
AbstractAutowireCapableBeanFactory.instantiate
-> SimpleInstantiationStrategy.instantiate
-> BeanUtils.instantiateClass
-> ctor.newInstance
populateBean
populateBean
是设置 Bean 属性的方法,如下图所示 autowireByName
和 autowireByType
两个方法即是自动注入的方法,以 autowireByName 为例,获取属性是以 getBean 的方式从 IOC 容器中获取对应的 Bean。
initializeBean
初始化 Bean 是在实例化之后的操作,在初始化之前和之后便是 BeanPostProcessor
的操作,初始化的操作便是 invokeInitMethods
的初始化方法。
# 在初始化之前和之后执行
applyBeanPostProcessorsBeforeInitialization
applyBeanPostProcessorsAfterInitialization
初始化 Bean 的操作
初始化之前和之后的操作方法:
循环依赖问题
循环依赖是绕不开的话题,循环依赖的问题具体的表现形式如下:
在讲循环依赖如何结果之前,还是涉及到 Bean 是如何创建的,如下图所示的过程就是解决循环依赖的过程。
1 在创建 A 对象时,需要在 populateBean 填充属性时触发获取 B 对象的操作,这里说一下会在 createBeanInstance 方法中将对象的构造方法放进 * 缓存中。
2 在经历了一轮 getBean 和 createBean 之后再次执行到属性赋值操作 populateBean,此时会再次触发获取 A 对象的操作,此时再去获取 A 对象时,会从 * 缓存中创建一个半成品 A 对象放进二级缓存中并删除 * 缓存,并做返回,此时 B 对象得到属性填充,完成赋值后放进一级缓存中,并将 B 对象返回到 1 步骤。
3 第一步的创建 A 对象继续,完成属性赋值后,会将对象放进一级缓存中,并删除二级缓存。 创建 Bean 的过程如下图所示,
Abstra ctAutowireCapableBeanFactory.doCreateBean
方法核心内容如下:
获取单例 Bean 的方法:
初始化的方法如下所示:
通过以上的三个步骤,就实现了循环依赖的问题解决,也完成了 Bean 对象的创建过程。
为什么要使用 * 缓存呢,说到底是要解决以下问题:
1 如果采用了一级缓存,如果没有存在循环依赖的问题,确实是可以的。如果有存在前图中的循环依赖问题,那么就无法解决了,就只能采用两级缓存才能解决了。
2 如果使用了两级缓存,确实能解决一部分的问题。但是 Bean 被 AOP 代理,再使用两级缓存就不能解决问题了,必须采用 * 缓存。
来源:https://juejin.cn/post/7083689894089850893


猜你喜欢
- 根据公司产品的要求,app要实现全屏模式,也就是4.4以后的所谓的沉浸式。在values-v19和values-v21的styles里添加以
- 本文实例讲述了C#图像处理之边缘检测(Smoothed)的方法。分享给大家供大家参考。具体如下://定义smoothed算子边缘检测函数pr
- Android指定SnackBar在屏幕的位置Snackbar 常以一个小的弹出框的形式,出现在手机屏幕下方或者桌面左下方,并且是在屏幕所有
- 快捷键是很多软件的常用功能,本文实例讲解了三种方法来实现C# button快捷键,如Alt + *(按钮快捷键),Ctrl+*及其他组合键等
- 本文研究的主要是Java数组的扩容的相关代码示例,具体实现过程如下所示。在写程序的过程中,我们常常会碰见数组空间不够用的情况,比如我已经初始
- 代理模式的应用:远程代理,为一个对象在不同的地址空间提供局部代表,可以隐藏一个对象存在于不同地质空间的事实。虚拟代理,根据需要创建开销很大的
- Spring简介1.什么是Springspring是分层的JavaSE及JavaEE应用于全栈的轻量级开源框架,以 IoC (Inverse
- 前言本文主要给大家介绍的是java虚拟机的故障处理工具,文中提到这些工具包括:名称主要作用jpsJVM process Status Too
- Spring 使用Junit单元测试并配置数据源一、问题描述由于公司项目中的数据源是配置在Tomcat中的server.xml中的,所以在使
- MyBatis提供了 * 接口,我们可以实现自己的 * ,将其作为一个plugin装入到SqlSessionFactory中。 首先要说的是
- 一、 ASCII码我们知道,在计算机内部,所有的信息最终都表示为一个二进制的字符串。每一个二进制位(bit)有0和1两种状态,因此八个二进制
- IDEA快速搭建spring boot项目1.创建项目老规矩,点击Create New Project2.编写控制器在com.demo.sp
- Configuration configuration = ConfigurationManager.OpenExeConfiguratio
- 环境搭建项目结构图:1.我们首先做好服务端pom.xml<dependencies>
- 本文实例为大家分享了Unity实现本地文本多语言化的具体代码,供大家参考,具体内容如下在unity项目过程中大多都会遇到多语言化,下面讲一下
- 1. 概述在这篇文章中,我们将使用Spring Boot实现一个基本的邮箱注册账户以及验证的过程。我们的目标是添加一个完整的注册过程,允许用
- 示例【通过班级查询老师信息】创建t_classes创建t_classessTeacher创建t_teacher创建Classespackag
- 最近的项目里用到了,在网上找不到合适的,于是自己写了个简单的,带回弹效果:可以自定义的属性有:<!-- 滑动解锁控件 xml配置属性
- 大家都知道在C#里面,我们可以使用Thread.Start方法来启动一个线程,当我们想停止执行的线程时可以使用Thread.Abort方法来
- 本文以案例形式分析了Android中TelephonyManager类的用法。分享给大家供大家参考。具体如下:目录结构:main.xml布局