Spring框架源码脉络分析(二):spring框架通过xml以及注解方式注册BeanDefinition的流程全链路分析


在上一章节中,主要介绍了SpringIoC、依赖注入和Spring中的Bean与BeanDefinition。可能部分读者还是比较迷茫,BeanDefinition到底是干什么用的,在本章节中,将通过利用Spring实例化Java对象的过程,先带领读者过一遍Spring框架注册BeanDefinition的运行流程,然后再深入其中,详细分析每一个步骤。

Spring注册BeanDefinition流程全链路简述

在Spring容器的实例化过程中,其核心流程其实非常的简单,逻辑也很清晰,主要是三个步骤:

  1. 解析配置
  2. 定位与注册对象
  3. 注入对象

其中,第三步注入对象将在IoC的最后一个章节进行分析,在本章节中主要分析实例化的前两个过程,因为这两个过程和BeanDefinition息息相关。

而在Spring中,前两个过程都整合在了一起,整个过程其实只是一个方法,位于AbstractApplicationContext类下的refresh()方法中的obtainFreshBeanFactory()方法中。refresh()方法又称为容器的刷新与初始化,在容器的刷新与初始化阶段中,一上来就对BeanDefinition进行了解析配置、定位与注册的过程。

在解析配置、定位与注册BeanDefinition的流程中,容器主要做的事情可以抽象出主要的三个阶段如下:
Spring框架源码脉络分析(二):spring框架通过xml以及注解方式注册BeanDefinition的流程全链路分析

Xml配置方式的资源定位、加载、解析、注册全链路分析

首先,在Spring源码中新建一个module,命名springdemo,然后仿照平时我们写业务的逻辑,创建一个Service层接口与实现类,然后写一个xml配置文件对该Service层接口进行配置,并在Entrance类中进行Xml配置实例化。
Spring框架源码脉络分析(二):spring框架通过xml以及注解方式注册BeanDefinition的流程全链路分析
Spring框架源码脉络分析(二):spring框架通过xml以及注解方式注册BeanDefinition的流程全链路分析
这里由于是xml配置形式,所以使用的容器实现类为FileSystemXmlApplicationContext,不熟悉容器类的请点击:容器类的具体介绍章节,这里主要让读者快速全局速览整个流程。

DefaultListableBeanFactory这个类中的registerBeanDefinition(String beanName, BeanDefinition beanDefinition) 方法打一个断点,这个方法是将BeanDefinition及其beanName注册进容器中。然后以Debug方式执行上述代码,不出意外代码会停在断点处。然后我们查看系统调用栈:
Spring框架源码脉络分析(二):spring框架通过xml以及注解方式注册BeanDefinition的流程全链路分析
此时,当我们利用ApplicationContext applicationContext = new FileSystemXmlApplicationContext(xmlPath);这个经典代码中注册BeanDefinition流程就清晰展示在了眼前。

在上面的系统调用栈中,其实refresh之后所有的方法执行,都还在refresh中,而且仅仅执行了前两行代码,只到obtainFreshBeanFactory()方法,当执行完obtainFreshBeanFactory()方法后,BeanDefinition就已经被注册进容器中了,所以refresh方法是IoC容器的核心部分,是容器加载的核心源码,在本章节中不深入进去分析,在本章节主要快速带领读者明白IoC容器流程脉络,主要在干什么,refresh方法具体分析可以看下一章节:容器的初始化与刷新逻辑

那么obtainFreshBeanFactory()到底在干什么呢?其实前文已经说得很明白了,这一阶段的核心是读取、解析与注册对象。下图把系统调用栈的流程进行了注释与分析。
Spring框架源码脉络分析(二):spring框架通过xml以及注解方式注册BeanDefinition的流程全链路分析
整个逻辑流程如下图所示:
Spring框架源码脉络分析(二):spring框架通过xml以及注解方式注册BeanDefinition的流程全链路分析