Bootstrap

Spring源码解析-Spring IOC

一:简述

这篇⽂章主要讲解 IOC 容器的创建过程,让你对整体有⼀个全局的认识,⽂章没有复杂嵌套的 debug 流程,相对 来说⽐较简单。

一. 基础知识

1.1 什么是 Spring IOC ?

IOC 不是⼀种技术,只是⼀种思想,⼀个重要的⾯向对象编程的法则,它能指导我们如何设计出松耦合、更优良的 程序。

传统应⽤程序都是由我们在类内部主动创建依赖对象,从⽽导致类与类之间⾼耦合,难于测试。

有了 IOC 容器后,把创建和查找依赖对象的控制权交给了容器,由容器进⾏注⼊组合对象,所以对象与对象之间是 松散耦合,便于测试和功能复⽤,整个体系结构更加灵活。

理解 IOC 的关键是要明确 “谁控制谁,控制什么,为何是反转(有反转就应该有正转了),哪些⽅⾯反转了”,我 们浅析⼀下:

  • 谁控制谁,控制什么
    • 传统 Java SE 程序设计,我们直接在对象内部通过 new 进⾏创建对象,是程序主动去创建依赖对象;
    • IOC 是有专⻔⼀个容器来创建这些对象,即由 IOC 容器来控制对象的创建;
    • 谁控制谁?当然是 IOC 容器控制了对象;
    • 控制什么?主要控制了外部资源获取。
  • 为何反转,哪些⽅⾯反转
    • 传统应⽤程序是由我们⾃⼰在对象中主动控制去直接获取依赖对象,也就是正转;
    • 反转则是由容器来帮忙创建及注⼊依赖对象;
    • 为何是反转?因为由容器帮我们查找及注⼊依赖对象,对象只是被动的接受依赖对象,所以是反转;
    • 哪些⽅⾯反转了?依赖对象的获取被反转了。

2.2 容器创建过程

在看源码之前,⾸先搞清楚 Spring IOC 容器的创建过程,然后再深⼊源码。

IOC 容器如婚姻介绍所:

  • 男⼈ A 要求需要⼀个⼥⼈,中介从婚姻介绍所根据男⼈ A 给他介绍了如花;
  • 男⼈ B 要求介绍⼀个肤⽩貌美⼤⻓腿,中介从婚姻介绍所根据男⼈ B 给他介绍了⼥神。

从婚姻介绍所的例⼦可以⼤概知道 Spring IOC 是存储、定制、管理等功能的各种定制化的 bean 对象容器,下图 是 Spring IOC 容器创建基本流程。

 

三. 核⼼知识

3.1 相关对象

3.1.1 ApplicationContext

ApplicationContext 接⼝是 BeanFactory 的⼦接⼝,也被称为 Spring 上下⽂,与 BeanFactory ⼀样,可以加载配 置⽂件中定义的 bean,并进⾏管理。

它还加强了企业所需要的功能,如从属性⽂件中解析⽂本信息和将事件传递给所有指定的监视器,下图是 ApplicationContext 接⼝的继承关系。

ApplicationContext 接⼝主要的 5 个作⽤如表所示:

3.1.2 BeanDefinitionReader

// 示例
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
复制代码

配置⽂件解析器,如将配置⽂件中的 bean 信息解析成对应的 BeanDefinition 对象。

xml ⽂件解析使⽤是就是BeanDefinitionReader 实现类 XmlBeanDefinitionReader。

3.1.3 BeanFactoryPostProcessor

可以修改 Spring 上下⽂中 BeanDefinition 信息。

如下图 BeanFactoryPostProcessor 的⼦类 PropertySourcesPlaceholderConfigurer 的作⽤可以为数据库连接池 ${} 占位符赋值等等。

3.1.4 BeanFactory

是所有 Bean 容器的根接⼝,定义了 spring 容器基本⽅法。

如使⽤ getBean(beanName,Class) 获取对象。

3.2 源码核⼼流程

容器初始化的核⼼源码,都在 refresh() ⽅法中:

public void refresh() throws BeansException, IllegalStateException {
    synchronized (this.startupShutdownMonitor) {
        //1:准备刷新上下⽂环境
        prepareRefresh();
        //2:获取初始化Bean⼯⼚
        ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
        //3:对bean⼯⼚进⾏填充属性
        prepareBeanFactory(beanFactory);
        try {
            //4:Spring开放接⼝ 留给⼦类去实现该接⼝
            postProcessBeanFactory(beanFactory);
            //5:调⽤我们的bean⼯⼚的后置处理器
            invokeBeanFactoryPostProcessors(beanFactory);
            //6:注册我们bean后置处理器
            registerBeanPostProcessors(beanFactory);
            //7:初始化国际化资源处理器
            initMessageSource();
            //8:初始化事件多播器
            initApplicationEventMulticaster();
            //9:这个⽅法同样也是留个⼦类实现,其中springboot也是从这个⽅法进⾏tomcat的启动
            onRefresh();
            //10:把我们的事件监听器注册到多播器上
            registerListeners();
            //11:实例化所有的⾮懒加载的单实例bean
            finishBeanFactoryInitialization(beanFactory);
            //12:最后刷新容器 发布刷新事件(Spring cloud eureka也是从这⾥启动的)
            finishRefresh();
        } catch (BeansException ex) {
            if (logger.isWarnEnabled()) {
                logger.warn("Exception encountered during context initialization -
                        " +
                        "cancelling refresh attempt: " + ex);
            }
            // Destroy already created singletons to avoid dangling resources.
            destroyBeans();
            // Reset 'active' flag.
            cancelRefresh(ex);
            // Propagate exception to caller.
            throw ex;
        } finally {
            // Reset common introspection caches in Spring's core, since we
            // might not ever need metadata for singleton beans anymore...
            resetCommonCaches();
        }
    }
}
复制代码

什么,内容不够细致?那我再给⼤家上⼀个⽜逼的思维导图:

四. 源码解析

下⾯主要对 refresh() 的 12 个流程进⾏讲解。

4.1 prepareRefresh()

准备刷新上下⽂环境:

protected void prepareRefresh() {
    // Switch to active.
    this.startupDate = System.currentTimeMillis();
    this.closed.set(false);
    this.active.set(true);
    if (logger.isInfoEnabled()) {
        logger.info("Refre
;