Spring IOC - Spring启动过程解析

Spring启动流程的核心逻辑主要体现在方法AbstractApplicationContext#refresh中,该方法没有被子类重写。

本文主要从宏观层面对其进行剖析,从整体上感知。各执行步骤和作用按先后顺序如下表所示,其中标红方法为核心方法,标绿色方法提供了扩展点。

|----------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 方法名 | 主要作用 |
| prepareRefresh() | 记录容器启动时间、设置容器状态;提供扩展点用于自定义添加key-value属性 |
| obtainFreshBeanFactory() | 获取一个新鲜的beanFactory实例,该方法由两个子类实现: 1.GenericApplicationContext:很简单,返回无参构造方法创建的beanFactory 2.AbstractRefreshableApplicationContext:关闭当前beanFactory,创建新的beanFactory,体现了刷新的含义 |
| prepareBeanFactory(beanFactory) | 提前给beanFactory塞入bean实例:如BeanFactory,ResourceLoader等,以及设置类加载器、属性编辑器等 |
| postProcessBeanFactory(beanFactory) | 此处是Spring留的扩展点,目的是在beanFactory准备好后,留给用户最后对beanFactory做一些设置 如对于Web应用,重写后给beanFactory塞入ServletContext等单例bean |
| invokeBeanFactoryPostProcessors(beanFactory) | 调用BeanFactoryPostProcessor的实现类对BeanFactory进行后置处理,如调用ConfigurationClassPostProcessor对常用注解进行解析,生成BeanDefinition |
| registerBeanPostProcessors(beanFactory) | 注册Bean的后置处理器,包括系统内置的和自定义的,如:AutowireAnnotationBeanPostProcessor、CommonAnnotationBeanPostProcessor |
| initMessageSource() | 初始化MessageSource,用于支持国际化 |
| initApplicationEventMulticaster() | 初始化事件派发器,用于支持事件监听 |
| onRefresh() | Spring提供的扩展点,留给子类实现 |
| registerListeners() | 注册事件监听器 |
| finishBeanFactoryInitialization(beanFactory) | 实例化所有非懒加载的单例Bean,Spring Bean的生命周期就是从这里开始的 |
| finishRefresh() | 完成刷新,发布容器刷新完成事件等 |

源码及注释如下:

java 复制代码
	// Spring应用上下文的核心方法,它负责完成Spring容器的初始化和刷新工作
	@Override
	public void refresh() throws BeansException, IllegalStateException {
		synchronized (this.startupShutdownMonitor) {
			// Prepare this context for refreshing.
			// 准备刷新操作,包括记录容器启动时间、设置容器状态、获取属性、验证必要的属性
			prepareRefresh();

			// Tell the subclass to refresh the internal bean factory.
			// 获取一个新鲜的BeanFactory实例,该方法由子类实现
			// 获取新的beanFactory,销毁原有beanFactory、为每个bean生成BeanDefinition等,注意,此处是获取新的,销毁旧的,这就是刷新的意义
			ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();

			// Prepare the bean factory for use in this context.
			// 对BeanFactory进行一些设置,例如设置类加载器、添加属性编辑器等
			prepareBeanFactory(beanFactory);

			try {
				// Allows post-processing of the bean factory in context subclasses.
				// 允许子类在BeanFactory标准初始化完成后对其进行进一步的处理
				// 此处是spring留的扩展点,目前是在beanFactory准备好后,留给用户最后对beanFactory做一些设置
				// 可参考GenericWebApplicationContext#postProcessBeanFactory实现
				postProcessBeanFactory(beanFactory);

				// Invoke factory processors registered as beans in the context.
				// 调用BeanFactoryPostProcessor的实现类对BeanFactory进行后置处理
				// 扫描解析的核心代码
				invokeBeanFactoryPostProcessors(beanFactory);

				// Register bean processors that intercept bean creation.
				// 注册BeanPostProcessor的实现类,用于在Bean创建过程中进行拦截
				registerBeanPostProcessors(beanFactory);

				// Initialize message source for this context.
				// 初始化MessageSource,用于支持国际化
				initMessageSource();

				// Initialize event multicaster for this context.
				// 初始化事件派发器,用于支持事件监听
				initApplicationEventMulticaster();

				// Initialize other special beans in specific context subclasses.
				// 留给子类实现,用于在容器刷新时进行特殊的处理
				onRefresh();

				// Check for listener beans and register them.
				// 注册事件监听器
				registerListeners();

				// Instantiate all remaining (non-lazy-init) singletons.
				// 实例化所有非懒加载的单例Bean
				// 这一步可谓和我们开发者打交道最多,我们自定义的Bean绝大多都是在这一步被初始化的,包括依赖注入等等
				// 因此了解这一步,能让我们更深入的了解Spring是怎么管理我们的Bean的生命周期,以及依赖关系的
				finishBeanFactoryInitialization(beanFactory);

				// Last step: publish corresponding event.
				// 完成刷新操作,包括发布容器刷新完成事件等
				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.
				// 如果在刷新过程中出现异常,会执行该方法取消操作,并销毁已经创建的单例Bean,最后,会重置Spring核心中的一些缓存
				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();
			}
		}
	}
相关推荐
北欧人写代码3 分钟前
javaWeb开发
java
犬余4 分钟前
设计模式之享元模式:看19路棋盘如何做到一子千面
java·设计模式·享元模式
老马啸西风5 分钟前
NLP 中文拼写检测纠正论文 Automatic-Corpus-Generation
java
时雨h5 分钟前
30天面试打卡计划 2024-12-25 26 27 面试题
java·开发语言·数据库
别致的影分身1 小时前
Linux 线程池
java·开发语言·jvm
山山而川粤3 小时前
母婴用品系统|Java|SSM|JSP|
java·开发语言·后端·学习·mysql
字节流动5 小时前
Android Java 版本的 MSAA OpenGL ES 多重采样
android·java·opengles
玉红7775 小时前
R语言的数据类型
开发语言·后端·golang
鹿屿二向箔6 小时前
基于SSM(Spring + Spring MVC + MyBatis)框架搭建一个病人跟踪信息管理系统
spring·mvc·mybatis
呜呼~225146 小时前
前后端数据交互
java·vue.js·spring boot·前端框架·intellij-idea·交互·css3