spring源码之bean的初始化及循环引用
实例化方法,把bean实例化,并且包装成BeanWrapper
1、点进这个方法里面。

这个方法是反射调用类中的 factoryMethod 方法。 这要知道@Bean 方法的原理, 实际上
spring 会扫描有@bean 注解的方法, 然后把方法名称设置到 BeanDefinition 的 factoryMethod
属性中, 接下来就会调到上面截图中的方法实现@Bean 方法的调用。
2、 有参构造函数的时候 
determineConstructorsFromBeanPostProcessors
这个方法是 BeanPostProcessor 接口类的首次应用, 最终会掉到
AutowiredAnnotationBeanPostProcessor 类的方法, 在方法中会扫描有注解的构造函数然后完
成装配过程。 然后把有有@Autowired 注解的构造函数返回。
3、 无参构造函数的实例化

这就是简单的反射实例化。 大部分类的实例化都会走这个逻辑
4、 类中注解的收集
实例化完成后接下来就需要对类中的属性进行依赖注入操作, 但是类里面属性和方法的依
赖注入往往用@Autowired 或者@Resource 注解, 那么这些注解的依赖注入是如何完成的
呢?
注解的收集:

也是通过 BeanPostProcessor 接口类型实例来挨个处理的。
A、 首先是
CommonAnnotationBeanPostProcessor 类, 这个类完成了@Resource 注解的属性或
者方法的收集
这个类还对@PostConstruct 和@PreDestory 支持

收集过程
1、 看缓存里面有没有 InjectionMetadata 对象
2、 从类中获取所有 Field 对象, 循环 field 对象, 判断 field 有没有@Resource 注解,
如果有注解封装成 ResourceElement 对象
3、 从类中获取所有 Method 对象, 循环 Method 对象, 判断 Method 有没有@Resource
注解, 如果有注解封装成 ResourceElement 对象
4、 最终把两个 field 和 Method 封装的对象集合封装到 InjectionMetadata 对象中
B、 然后是
AutowiredAnnotationBeanPostProcessor 类, 对@Autowired 注解的属性和方法
的收集。 收集过程基本上跟@Resource 注解的收集差不多, 这里就不赘述了
5、 IOC\DI 依
对应的方法:


这里又是一个 BeanPostProcessor 类型接口的运用, 前面我们讲到了@Resource@Autowired 注解的收集, 那么这个方法就是根据收集到的注解进行反射调用。

循环收集到的 metaData 中的 list 对象, 然后挨个调用里面的 InjectedElement 的inject 方法完成依赖注入。

其中 value 值的获取, 如果依赖的属性是一个引用类型必定会触发该属性的
BeanFactory.getBean 操作, 从而从 spring 容器中获取到对应的实例。 方法的依赖注
入类似这里就不再赘述。
6、 bean 实例化后的操作
代码走到这里:

A、 首先是对某些 Aware 接口的调用

B、 然后@PostConstruct 注解方法的调用

这里又是一个 BeanPostProcessor 接口的运用,
前面讲过, 有@PostConstruct 注解的方法会收集到一个 met
就是通过 BeanPostProcessor 接口调到
CommonAnnotationBeanPostProcessor 类, 然后在类中拿
根据对象里面的容器来反射调用有注解的方法。 代码如下:


有@PostConstruct 注解的容器会收集到 initMethods 容器中, 接下来就是方法的
反射调用。

C、 InitializingBean 接口和 init-method 属性调用

Init-method 属性调用是在 afterPropertiesSet 之后

afterPropertiesSet和Init-method和有@PostConstruct注解的方法其实核
心功能都是一样的, 只是调用时序不一样而已, 都是在该类实例化和 IOC 做完后调用
的, 我们可以在这些方法中做一些在 spring 或者 servlet 容器启动的时候的初始化
工作。 比如缓存预热, 比如缓存数据加载到内存, 比如配置解析, 等等初始化工作。
在这个方法里面还有一个重要的逻辑

也是一个 BeanPostProcessor 接口的运用, 在这里会返回 bean 的代理实例, 这个
就是 AOP 的入口。
D、 FactoryBean 接口
带入如下:

在实例化和 IOC/DI 做完后, 就会调用 FactoryBean 类型的接口, 如果要获取到
FactoryBean 类本身, 就必须加上”&”符号, 比如
beanFactory.getBean(“&beanName” )。

BeanFactory.getBean(“beanName” )只能获取到 getObject()方法返回的实
例。
getObject 方法返回的实例会有单独的缓存存储, 跟其他实例不是同一个缓存, 对应的缓
存是: factoryBeanObjectCache
E、 循环依赖
循环依赖请参照流程图理解
https://www.processon.com/view/link/5df9ce52e4b0c4255ea1a84f
循环依赖只会出现在单例实例无参构造函数实例化情况下
有参构造函数的加@Autowired 的方式循环依赖是直接报错的, 多例的循环依赖也是
直接报错的


循环依赖步骤:
1、 A 类无参构造函数实例化后, 设置三级缓存
2、 A 类 populateBean 进行依赖注入, 这里触发了 B 类属性的 getBean 操作
3、 B 类无参构造函数实例化后, 设置三级缓存
4、 B 类 populateBean 进行依赖注入, 这里触发了 A 类属性的 getBean 操作
5、 A 类之前正在实例化, singletonsCurrentlyInCreation 集合中有已经
有这个 A 类了, 三级缓存里面也有了, 所以这时候是从三级缓存中拿到的提前暴露的
A 实例, 该实例还没有进行 B 类属性的依赖注入的, B 类属性为空。
6、 B 类拿到了 A 的提前暴露实例注入到 A 类属性中了
7、 B 类实例化已经完成, B 类的实例化是由 A 类实例化中 B 属性的依赖注入触发
的 getBean 操作进行的, 现在 B 已经实例化, 所以 A 类中 B 属性就可以完成依赖注
入了, 这时候 A 类 B 属性已经有值了
8、 B 类 A 属性指向的就是 A 类实例堆空间, 所以这时候 B 类 A 属性也会有值了。
spring源码之bean的初始化及循环引用的更多相关文章
- Spring源码-IOC部分-容器初始化过程【2】
实验环境:spring-framework-5.0.2.jdk8.gradle4.3.1 Spring源码-IOC部分-容器简介[1] Spring源码-IOC部分-容器初始化过程[2] Spring ...
- Spring源码系列 — Bean生命周期
前言 上篇文章中介绍了Spring容器的扩展点,这个是在Bean的创建过程之前执行的逻辑.承接扩展点之后,就是Spring容器的另一个核心:Bean的生命周期过程.这个生命周期过程大致经历了一下的几个 ...
- Spring源码系列 — 构造和初始化上下文
探索spring源码实现,精华的设计模式,各种jdk提供的陌生api,还有那么点黑科技都是一直以来想做的一件事!但是读源码是一件非常痛苦的事情,需要有很大的耐心和扎实的基础. 在曾经读两次失败的基础上 ...
- Spring源码之Bean生命周期
https://www.jianshu.com/p/1dec08d290c1 https://www.cnblogs.com/zrtqsk/p/3735273.html 总结 将class文件加载成B ...
- Spring 源码学习 04:初始化容器与 DefaultListableBeanFactory
前言 在前一篇文章:创建 IoC 容器的几种方式中,介绍了四种方式,这里以 AnnotationConfigApplicationContext 为例,跟进代码,看看 IoC 的启动流程. 入口 从 ...
- Spring源码 21 Bean生命周期
参考源 https://www.bilibili.com/video/BV1tR4y1F75R?spm_id_from=333.337.search-card.all.click https://ww ...
- 【spring源码】bean的实例化(转载)
首先来看一段代码,看过上一节的朋友肯定对这段代码并不陌生.这一段代码诠释了Spring加载bean的完整过程,包括读取配置文件,扫描包,加载类,实例化bean,注入bean属性依赖. 上一节介绍了Sp ...
- Spring源码学习
Spring源码学习--ClassPathXmlApplicationContext(一) spring源码学习--FileSystemXmlApplicationContext(二) spring源 ...
- Spring源码-IOC部分-Spring是如何解决Bean循环依赖的【6】
实验环境:spring-framework-5.0.2.jdk8.gradle4.3.1 Spring源码-IOC部分-容器简介[1] Spring源码-IOC部分-容器初始化过程[2] Spring ...
随机推荐
- mysql事务级别和spring中应用
一.事务的基本要素(ACID) 1.原子性(Atomicity):事务开始后所有操作,要么全部做完,要么全部不做,不可能停滞在中间环节.事务执行过程中出错,会回滚到事务开始前的状态,所有的操作就像没有 ...
- quartz表(11张)
/* Navicat Premium Data Transfer Source Server : 本地连接 Source Server Type : MySQL Source Server Versi ...
- synchronized的实现原理——对象头解密
前言 并发编程式Java基础,同时也是Java最难的一部分,因为与底层操作系统和硬件息息相关,并且程序难以调试.本系列就从synchronized原理开始,逐步深入,领会并发编程之美. 正文 基础稍微 ...
- 【Android】Android开发可以手动进行控制的跑马灯效果,包括从左到右,以及从右到左,
作者:程序员小冰,GitHub主页:https://github.com/QQ986945193 新浪微博:http://weibo.com/mcxiaobing 首先给大家看一下我们今天这个最终实现 ...
- shader之间的数据传递
shader之间传递数据实在是太常用了. 下面我们总结几种shader之间传递数据的方法. Name based matching 最简单,也是最常用的一种传递方式是依靠名字进行匹配. 例如我们从ve ...
- adb 打开手机端口进行无线模式调试
打开手机端口 让手机在指定的端口可以接收到TCP/IP连接. 确保手机开启了usb调试 用usb线把手机和电脑连接起来 执行命令:adb tcpip 5555 执行成功后就可以把usb线拔掉了,端口可 ...
- 几个Graphics函数
1.Graphics.Blit:Copies source texture into destination render texture with a shader 声明: 1.public sta ...
- 深入了解Redis【一】源码下载与参考资料准备
引言 一直在使用redis,但是却没有系统的了解过它的底层实现,准备边学习边记录,深入了解redis. 打算分析以下几个方面: redis的基本类型及底层原理与java对比,每种数据类型的使用场景 r ...
- 13_Python的面向对象编程-类class,对象object,实例instance
1.面向对象概述 1.类是用来描述对象的工具,把拥有相同属性和行为的对象分为一组 2.对象是由类实例化出来的一个具体的对象 属性: 对象拥有的名词,用变量表示 ...
- Git 实用操作:重写 Commit 历史
当我们修改完代码,提交了一个 commit,然后发现改错了,怎么修正?下面分两种情况来讨论:修正最近一次提交,和修正历史多个提交. 修正最近一次提交 如果发现刚刚提交的内容有错误,当场再修改一下再提交 ...