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() | 完成刷新,发布容器刷新完成事件等 |
源码及注释如下:
- // 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();
- }
- }
- }