Spring启动过程中创建bean过程中初始化前部分的代码分析
实例化前的作用
1.初始化前,也是Spring提供的一个扩展点:BeanPostProcessor.postProcessBeforeInitialization()。
2.利用初始化前,可以对进行了依赖注入的Bean进行处理。
使用示例
1.首先必须要添加@Component注解,因为处理器不加入容器是不会被使用的
2.被调用时,这里的bean是已经被实例化而且已经注入属性了(故可以看做是一个完整的实例对象),其次这个处理器是每个Bean对象都要调用的,如果处理只是针对某个Bean,或者某一类Bean,其实是要做区分的。
@Component public class BeanPostProcessorDemo implements BeanPostProcessor { @Override public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { if ("userService".equals(beanName)){ System.out.println("初始化前"); } return bean; } }
实例化前逻辑代码展示(这段代码位于 doCreateBean方法#initializeBean(beanName, exposedObject, mbd)方法#applyBeanPostProcessorsBeforeInitialization方法)
public Object applyBeanPostProcessorsBeforeInitialization(Object existingBean, String beanName) throws BeansException { Object result = existingBean; for (BeanPostProcessor processor : getBeanPostProcessors()) { Object current = processor.postProcessBeforeInitialization(result, beanName); if (current == null) { return result; } result = current; } return result; }
代码说明
1.为什么是要遍历全部的BeanPostProcessor,而不是抽出需要的部分形成缓存?
这里面涉及到了接口的继承关系,BeanPostProcessor是顶级接口,也就是最上层的,其余的用于实例化接口InstantiationAwareBeanPostProcessor,合并BeanDefinition的接口MergedBeanDefinitionPostProcessor都是他的子接口,子接口本身会继承父接口的方法,故也不能排除开发者会在其中实现初始化前的方法。所以才会全部拿出来遍历。不过毕竟这些接口都是存于缓存 AbstractBeanFactory类#private final List<BeanPostProcessor> beanPostProcessors = new CopyOnWriteArrayList<>();属性中。
2.在Spring源码中的调用:
1)InitDestroyAnnotationBeanPostProcessor会在初始化前这个步骤中执行@PostConstruct的方法。(可查看 注解@PostConstruct分析)
2)ApplicationContextAwareProcessor会在初始化前这个步骤中进行其他Aware的回调:
1. EnvironmentAware:回传环境变量
2. EmbeddedValueResolverAware:回传占位符解析器
3. ResourceLoaderAware:回传资源加载器
4. ApplicationEventPublisherAware:回传事件发布器
5. MessageSourceAware:回传国际化资源
6. ApplicationStartupAware:回传应用其他监听对象,可忽略
7. ApplicationContextAware:回传Spring容器ApplicationContext