一文读懂Spring框架中依赖注入流程

Posted 小杰

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了一文读懂Spring框架中依赖注入流程相关的知识,希望对你有一定的参考价值。

想读懂Spring的依赖注入流程,我们先简单了解一下Ioc和DI是什么?

IoC和DI

  • Ioc—Inversion of Control,即“控制反转”,不是什么技术,而是一种设计思想
  • Ioc就是将你设计好的对象交给容器来进行管理
  • DI—Dependency Injection,即“依赖注入”:组件之间依赖关系由容器在运行期决定
  • DI就是由容器动态的将某个依赖关系注入到组件之中
  • 简单来说就是不通过new()的方式在类内部创建依赖对象,而是将依赖的类对象在外部创建好通过构造方法,函数参数等方式传递给类使用
  • 依赖注入的方式有3种:
  • 接口注入(调用和实现分离):不常用
  • setter注入 (使用set方法为属性赋值)
  • 构造器注入(通过构造方法赋值)(不能解决循环依赖的问题)

 

依赖注入流程图:

 

根据依赖注入流程图简单解释:

1.依赖注入的步骤在Bean实例化之后就进行依赖注入了:可以根据Bean的生命周期流程图可知:

2.依赖注入入口:AbstractAutowireCapableBeanFactory的docreateBean()方法中的populateBean(beanName, mbd, instanceWrapper);

protected Object doCreateBean(final String beanName, final RootBeanDefinition mbd, final @Nullable Object[] args)
            throws BeanCreationException {
// Initialize the bean instance.
        Object exposedObject = bean;
        try {
        // bean初始化第二步:填充属性(DI依赖注入发生在此步骤)
            // 调用反射和内省去进行属性设置
            // 属性值需要进行类型转换,判断属性注入是否需要
            populateBean(beanName, mbd, instanceWrapper);
            // bean初始化第三步:调用初始化方法,完成bean的初始化操作(AOP发生在此步骤)
            exposedObject = initializeBean(beanName, exposedObject, mbd);
        }
}

3.先进行一个bean的属性值的类型的转换(就是配置文件->依赖注入的类型) 转换之前存储的是 MutablePropertyValue(List<PropertyValue>) 

 3.1.然后中间经过BeanDefinitionValueResolve来进行转换:

3.2.然后继续存入 MutablePropertyValue(List<PropertyValue>)中:

4.Bean的属性注入(BeanWrapperImp) 先从PropertyAccessor(属性访问器)获取属性名称 然后获取指定的属性访问器去操作属性的注入:

 AbstractNestablePropertyAccessor的getPropertyAccessorForPropertyPath()的内部处理来获取属性访问器:

流程是先通过AbstractNestablePropertyAccessor的setpropertyValue()方法:

PropertyTokenHolder是什么?

  • 用于解析嵌套属性名称,标识唯一的属性
  • 解析后将 [] 之间的 \' 和 " 去除后保存在 canonicalName 中,attrs 保存在 actualName 中,["key", "0"] 保存在 keys 中

 再判断PropertyTokenHolder的keys()为不为空是进行其他属性注入还是普通属性注入:

 最后属性注入的核心逻辑是通过AbstractNestablePropertyAccessor的setPropertyValue()来进行注入逻辑的:

 不要白嫖点个赞 谢谢了!!!!

一文读懂Spring框架中Bean的生命周期

我们先来聊聊bean的生命周期:

bean的生命周期图:

AbstractAutowireCapableBeanFactory的docreateBean()方法(简单描述):

 1.可以根据源码的时候得出在docreateBean方法里面是bean生命周期的核心逻辑:

//只放了核心逻辑   
protected Object doCreateBean(final String beanName, final RootBeanDefinition mbd, final @Nullable Object[] args) throws BeanCreationException { // bean初始化第一步:默认调用无参构造实例化Bean // 构造参数依赖注入,就是发生在这一步 if (instanceWrapper == null) { //再这个地方完成了对象的创建,推断创建构造方法 进入 instanceWrapper = createBeanInstance(beanName, mbd, args); } // Initialize the bean instance. Object exposedObject = bean; try { // bean初始化第二步:填充属性(DI依赖注入发生在此步骤) // 调用反射和内省去进行属性设置 // 属性值需要进行类型转换,判断属性注入是否需要 populateBean(beanName, mbd, instanceWrapper); // bean初始化第三步:调用初始化方法,完成bean的初始化操作(AOP发生在此步骤) exposedObject = initializeBean(beanName, exposedObject, mbd); } }

我们根据Bean生命周期图进行Bean生命周期的内部讲解:

详细步骤介绍:

实例化:

1:Bean的实例化调用createBeanInstance(beanName, mbd, args)方法默认调用无参构造实例化Bean

依赖注入:

2.Bean的依赖注入  ->调用->populateBean(beanName, mbd, instanceWrapper)方法完成 Bean 中所有属性值的配置注入,最终核心还是根据PropertyHandler的setValue()方法进行setter方法的调用(注入);

初始化:

3.检查有无Aware的相关接口设置相关依赖:

3.1.如果Bean 实现了 BeanNameAware 接口,则 Spring 调用 Bean 的 setBeanName() 方法传入当前 Bean 的 id 值。

3.2.如果 Bean 实现了 BeanFactoryAware 接口,则 Spring 调用 setBeanFactory() 方法传入当前工厂实例的引用。

3.3.如果 Bean 实现了 ApplicationContextAware 接口,则 Spring 调用 setApplicationContext() 方法传入当前 ApplicationContext 实例的引用。

.

4.如果 BeanPostProcessor 和 Bean 关联,spring将调用它们的postProcessBeforeInitialization(Object bean, String beanName)(预初始化)方法它作用是在Bean实例创建成功后进行增强处理(如对Bean进行修改,增加某个功能等)。

5.是否实现InitializingBean接口,如果 Bean 实现了 InitializingBean 接口,则 Spring 将调用 afterPropertiesSet() 方法作用是在bean正式构造完成前增加我们自己自定义的逻辑(只能增加一些额外的逻辑)。

6.是否配置自定义的init方法(比如说配置文件中通过 init-method 属性指定了初始化方法),配置则执行初始化方法。

7.如果 BeanPostProcessor 和 Bean 关联,spring将调用它们的postProcessBeforeInitialization(Object bean, String beanName),(预初始化)作用和步骤4一样,只是时机不一样,步骤4是初始化前,这是初始化后。

销毁步骤:

8.如果 Bean 实现了 DisposableBean 接口,则 Spring 会调用 destory() 方法将 Spring 中的 Bean 销毁

9.如果在配置文件中通过 destory-method 属性指定了 Bean 的销毁方法,则 Spring 将调用该方法对 Bean 进行销毁

总结:

Bean的生命周期基本上分为4大步:

1.Bean的实例化 ->createBeanInstance()

2.Bean的属性注入(依赖注入)  ->populateBean(beanName, mbd, instanceWrapper);

3.Bean的初始化: ->initializeBean(beanName, exposedObject, mbd);

 4:Bean的销毁:是容器关闭时调用的,在ConfigurableApplicationContext类的close()中

以上是关于一文读懂Spring框架中依赖注入流程的主要内容,如果未能解决你的问题,请参考以下文章

Spring框架进阶Spring V3.0 DI源码分析流程

spring练习,在Eclipse搭建的Spring开发环境中,使用set注入方式,实现对象的依赖关系,通过ClassPathXmlApplicationContext实体类获取Bean对象(代码片段

10分钟学Spring:一文搞懂spring依赖注入(DI)

Spring框架--依赖注入

Spring框架核心功能手写实现

干货|一文读懂 Spring Data Jpa!