Pre
接上文 Spring5源码 - 03 普通对象对应的BeanDefinition是如何存入DefaultListableBeanFactory#beanDefinitionMap 源码分析
refresh()
这里我们只粗略的看一下其中的逻辑,真的很复杂
@Override public void refresh() throws BeansException, IllegalStateException { synchronized (this.startupShutdownMonitor) { StartupStep contextRefresh = this.applicationStartup.start("spring.context.refresh"); // Prepare this context for refreshing. 1:准备刷新上下文环境 prepareRefresh(); // Tell the subclass to refresh the internal bean factory. 2:获取告诉子类初始化Bean工厂 ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory(); // Prepare the bean factory for use in this context. 3:对bean工厂进行填充属性 prepareBeanFactory(beanFactory); try { // Allows post-processing of the bean factory in context subclasses. 第四:留给子类去实现该接口 postProcessBeanFactory(beanFactory); StartupStep beanPostProcess = this.applicationStartup.start("spring.context.beans.post-process"); // Invoke factory processors registered as beans in the context. 调用bean工厂的后置处理器. invokeBeanFactoryPostProcessors(beanFactory); // Register bean processors that intercept bean creation. 调用bean的后置处理器 registerBeanPostProcessors(beanFactory); beanPostProcess.end(); // Initialize message source for this context. 初始化国际化资源处理器. initMessageSource(); // Initialize event multicaster for this context. 创建事件多播器 initApplicationEventMulticaster(); // Initialize other special beans in specific context subclasses. 这个方法同样也是留给子类实现的 springboot也是从这个方法进行启动tomat的. onRefresh(); // Check for listener beans and register them. 把我们的事件监听器注册到多播器上 registerListeners(); // Instantiate all remaining (non-lazy-init) singletons. 实例化我们剩余的单实例bean. finishBeanFactoryInitialization(beanFactory); // Last step: publish corresponding event. 最后容器刷新 发布刷新事件(Spring cloud也是从这里启动的) 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(); contextRefresh.end(); } } }
Spring的设计
我们接着来看 spring 是在哪一个方法调用中完成扫描的 即什么时候初始化BeanDefinition, 并且把这个BeanDefinition放到bdMap集合中。
通过debug分析可以得出
invokeBeanFactoryPostProcessors(beanFactory);
主要工作
执行所有的需要被执行的BeanFactoryPostProcessor
1.1 执行Spring内置的BeanFactoryPostProcessor (完成Bean的扫描) 【其实就是ConfigurationClassPostProcessor】
1.2 执行开发人员提供的BeanFactoryPostProcessor
源码验证
再细说一下主要的设计思想
看方法名 invokeBeanFactoryPostProcessors 也能猜到 主要是执行 BeanFactoryPostProcessors
既然是 BeanFactoryPostProcessors , 那就说明有很多 BeanFactoryPostProcessor
BeanFactoryPostProcessor是个接口,spring实现了一部分,当然开发人员也可以实现BeanFactoryPostProcessor接口。
那Spring该如何决定这些接口实现类的先后顺序呢? 这就是Spring IOC 核心的地方。
先说说Spring的几个比较核心的子类和实现类
主要干的两个活
- 执行直接实现了BeanFactoryPostProcessor
- 执行实现了BeanDefinitionRegistryPostProcessor
举个例子 ,假设我们有个自己的类 ,实现了 子类BeanDefinitionRegistryPostProcessor接口
ArtisanTest02 implements BeanDefinitionRegistryPostProcessor
那么你不仅要把BeanDefinitionRegistryPostProcessor的接口实现,还要实现其父类BeanFactoryPostProcessor接口的方法,那么在调用invokeBeanFactoryPostProcessors的时候 ,这两个方法都会被执行 。
假设还有一个类 ,ArtisanTest03 实现了 BeanFactoryPostProcessor 接口
ArtisanTest03 implements BeanFactoryPostProcessor
来运行看下效果