Spirng源码全系列:
Spring IOC源码:简单易懂的Spring IOC 思路介绍
Spring IOC源码:核心流程介绍
Spring IOC源码:ApplicationContext刷新前准备工作
Spring IOC源码:obtainFreshBeanFactory 详解(上)
Spring IOC源码:obtainFreshBeanFactory 详解(中)
Spring IOC源码:obtainFreshBeanFactory 详解(下)
Spring IOC源码:<context:component-scan>源码详解
Spring IOC源码:invokeBeanFactoryPostProcessors 后置处理器详解
Spring IOC源码:registerBeanPostProcessors 详解
Spring IOC源码:实例化前的准备工作
Spring IOC源码:finishBeanFactoryInitialization详解
Spring IoC源码:getBean 详解
Spring IoC源码:createBean( 上)
Spring IoC源码:createBean( 中)
Spring IoC源码:createBean( 下)
Spring IoC源码:finishRefresh 完成刷新详解
前言
学习Spring也有一段时间了,后面写写文章来总结跟巩固对Spring IOC的理解,这篇文章也能给打算学习Spring IOC的初学者介绍一下整体的流程,看过源码的可以复习一下知识。本人学习Spring源码基于5.2.0,知识点凭自己理解,如有理解错的地方,希望可以交流共同进步。
核心概念
1. IOC(Inversion of Control)控制反转
使用对象时,由主动new产生对象转换成,从外部提供对象,在这个过程中,对象的创建控制权由程序转移到外部,此思想称为控制反转。
其实就是以前自己手动 new 对象,现在交给Spring。在项目启动加载时,交由Spring 来帮我们统一创建,需要用时直接从Spring 缓存中拿,这里取出来的对象是单例或者原型,是否懒加载等都是由我们自己提前配置好。
2. DI(Dependency Injection)依赖注入
在容器中建立bean与bean之间的依赖关系的整个过程,成为依赖注入。
我们平时创建的对象一般都会有属性值,Spring既然帮我们创建了对象了,但是创建出来的对象其实是半成品,因为对象的属性值都还是为空呢,这里的DI就是为对象属性赋值,但是赋值的前提是所需要的属性依赖也有交给Spring管理才行,注入时Spring会根据当前对象的属性去从三级缓存中查询是否存在,有则赋值。
流程概述
1.流程设计
如果让大家设计一个IOC容器,你会如何设计?
通过以上步骤可以设计出一个简陋版的IOC,一个Bean在Spring IOC中创建思路也是类似这样,但是要复杂得多。Spring拓展性设计的非常好,它提供了很多拓展接口,比如让使用者可以对配置文件解析进行操作的BeanFactoryPostProcessor,也有初始化前后执行的BeanPostProcessor。
2.Spring IOC流程
Spring时序图看起来就要复杂得多,比较人家是个框架,就要考虑到拓展性,方便使用者可以自由进行拓展,AOP也是IOC的一个拓展,它其实就是在Bean属性初始化后,IOC会执行BeanPostProcessor,AOP的逻辑就在这个拓展类中的After步骤实现的。像我们现在开发基本都是使用注解的,注解也是基于配置文件的基础之上,步骤基本都一样,就是在解析配置的处理不太一样,最终封装的配置文件对象都是一样的。
3.常见对象
Bean
在 Spring 中,构成应用程序主干并由 Spring IoC 容器管理的对象称为 bean。 bean 是一个由 Spring IoC 容器实例化,组装和管理的对象。
我们可以理解为交给IOC进行管理的对象就是Bean对象,像XML配置里面的Bean、常用的注解@component、@Service、@Bean等相当于告诉Spring IOC这些类你来帮我创建,而像@Autowire 相当于告诉IOC容器需要帮在三级缓存中找出我需要的对象并帮我进行注入赋值。
BeanDefinition
BeanDefinition就是将我们的配置文件或者注解扫描进行解析,将这些解析后的各种属性封装成的对象,为后续实例化初始化做准备。
BeanFactory 和 ApplicationContext
BeanFactory :可以理解成我们常说的IOC,像BeanDefinition、半成品缓存对象、成品缓存对象都是放在这里面的。
ApplicationContext:BeanFactory 的子接口,在 BeanFactory 的基础上构建,是相对比较高级的 IoC 容器实现。包含 BeanFactory 的所有功能,还提供了其他高级的特性,比如:事件发布、国际化信息支持、统一资源加载策略等。正常情况下,我们都是使用的 ApplicationContext。
以电话来举例:
我们家里使用的 “座机” 就类似于 BeanFactory,可以进行电话通讯,满足了最基本的需求。
而现在非常普及的智能手机,iPhone、小米等,就类似于 ApplicationContext,除了能进行电话通讯,还有其他很多功能:拍照、地图导航、听歌等。
FactoryBean
一般情况下,我们将 bean 的创建和管理都交给 Spring IoC 容器,Spring 会利用图中的流程来进行对象的创建,但是如果我们想自己实现 bean 的创建操作,可以实现吗?答案是可以的,FactoryBean 就可以实现这个需求。
FactoryBean 是一种特殊的 bean,它是个工厂 bean,可以自己创建 bean 实例,如果一个类实现了 FactoryBean 接口,则该类可以自己定义创建实例对象的方法,只需要实现它的 getObject() 方法即可。
FactoryBean 可能对于普通开发来说基本用不到也没去注意过,但是它其实应用的非常广,特别是在中间件中,如果你看过一些中间件的源码,一定会看到 FactoryBean 的身影。
总结
通过上面的介绍,我们对IOC应该不会一无所知的,它的流程:
1、加载配置文件
2、解析配置文件
3、封装配置文件
4、调用执行实现拓展接口BeanFactoryPostProcessor的类
5、注册多播器
6、实例化
7、初始化(初始化前后会执行实现beanPostProcessor的类)
8、销毁
这里总结得并不齐全,很多细节方法都省略了,主要就是要体现大概的流程,具体还得看源码深入