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();
			}
		}
	}
相关推荐
24k小善3 分钟前
FlinkSql入门与实践
java·大数据·flink·云计算
CodeCraft Studio17 分钟前
Excel处理控件Spire.XLS系列教程:Java设置Excel活动工作表或活动单元格
java·python·excel
余渔鱼112330 分钟前
ASP.NET CORE部署IIS的三种方式
后端·github·asp.net
瓯雅爱分享41 分钟前
任务管理系统,Java+Vue,含源码与文档,科学规划任务节点,全程督办保障项目落地提效
java·mysql·vue·软件工程·源代码管理
chxii41 分钟前
2.3java运算符
java
余辉zmh1 小时前
【Linux系统篇】:信号的生命周期---从触发到保存与捕捉的底层逻辑
android·java·linux
小布不吃竹1 小时前
Maven的概念与初识Maven
java·maven
中东大鹅1 小时前
Maven进阶
java·maven
嘻嘻嘻嘻嘻嘻ys1 小时前
《Spring Boot 3 + GraalVM原生镜像实战:云原生时代的毫秒启动与性能调优》
前端·后端
嘻嘻嘻嘻嘻嘻ys1 小时前
《Spring Boot 3.0×GraalVM:云原生时代的毫秒级启动实战革命》
前端·后端