spring 是如何注入对象的和bean 创建过程分析
sanri1993 人气:0文章目录:
- beanFactory 及 bean 生命周期起步
- BeanFactory refresh 全过程
- BeanFactoryPostProcessor 和 BeanPostProcessor 解析
- 使用 BeanPostProcessor 实现 aop 和 springboot Cache 相关注解实现
- 【本文】spring 是如何注入对象的
首先需要知道一个大致实现
这个注入过程肯定是在
BeanPostProcessor
中实现的spring 是在
beanFactory.getBean
进行 bean 实例化的,即懒加载根据第二条,也就是说在 getBean 的时候才会去调用所有
BeanPostProcessor
第二篇文章说到,BeanFactory 的 refresh 过程只是注册
BeanPostProcessor
,真正执行在 getBean 方法中MergedBeanDefinitionPostProcessor
也是一种BeanPostProcessor
它是专门用来处理注入的,如@Autowired
和@Value
,它重新弄了个一个生命周期函数,替代了BeanPostProcessor
默认的生命周期函数,这么看吧,我贴一小段源码for (BeanPostProcessor bp : getBeanPostProcessors()) { if (bp instanceof MergedBeanDefinitionPostProcessor) { MergedBeanDefinitionPostProcessor bdp = (MergedBeanDefinitionPostProcessor) bp; bdp.postProcessMergedBeanDefinition(mbd, beanType, beanName); } }
它允许你在非
BeanFactoryProcess
中去修改 Bean 定义
@Autowired
加载定义的过程
我们先不看 bean 的创建过程,就看 MergedBeanDefinitionPostProcessor
的实现子类,这里看名字猜测 AutowiredAnnotationBeanPostProcessor
应该就是干这件事的,所以我们接下来可以直接看 AutowiredAnnotationBeanPostProcessor 的 postProcessMergedBeanDefinition 方法的代码。
顺着方法的调用,可以知道在 buildAutowiringMetadata
是真正查找这些注解的地方,最后 checkConfigMembers
将 Member
注册进了 bean 定义,具体如何查找的读者自行查看源码。
这里只是将 Member
注册进了 bean 定义,真正实例化在填充 Bean 的过程中,下面说到 bean 的创建过程可以知道是何时注入的。
Bean 的创建过程
前面说到 spring 是在 getBean 的过程中进行 Bean 创建的,创建 bean 分为几个步骤
- 获取 bean 定义
- new Bean()
- 执行生命周期函数 (前)
- 创建依赖项
- 填充 bean
- 执行生命周期函数(后)
入口为 BeanFactory.getBean ,BeanFactory 的实现类为 DefaultListableBeanFactory
这些你可以在 BeanFactory 的 refresh 过程中找到
根据源码,如果 bean 还不存在时,就会执行 bean 的创建流程
获取 bean 定义在这段源码中
final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);
紧跟着,根据 Bean 定义搜索其依赖项,并创建 bean ,可以看出是递归创建 bean
String[] dependsOn = mbd.getDependsOn();
for (String dep : dependsOn) {
getBean(dep);
}
然后就创建 bean 了
if (mbd.isSingleton()) {
createBean(beanName, mbd, args);
}
// 真正的执行在 doCreateBean 过程中
Object beanInstance = doCreateBean(beanName, mbdToUse, args);
创建 bean 第一步 new Bean
if (instanceWrapper == null) {
instanceWrapper = createBeanInstance(beanName, mbd, args);
}
创建 bean 第二步,执行所有的 processor ,包含 MergedBeanDefinitionPostProcessor
,所以在这一步注册注入选项
applyMergedBeanDefinitionPostProcessors(mbd, beanType, beanName);
创建 bean 第三步,填充 bean ,这里做的 @Autowired
注入
populateBean(beanName, mbd, instanceWrapper);
深入源码可以看到这个
// 这里就是常说的根据名称注入,根据类型注入了
if (mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_BY_NAME) {
autowireByName(beanName, mbd, bw, newPvs);
}
if (mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_BY_TYPE) {
autowireByType(beanName, mbd, bw, newPvs);
}
因为在前面已经获取过依赖项,并且把其丢进了容器,所以这里是直接用反射写进去就可以了
创建 bean 第四步,初始化 bean ,这里有一个方法注入,方法注入原来发生在初始化 bean 过程中,还有就是生命周期函数执行了,包含 BeanPostProcessor
的前置后置生命周期,初始化方法等
一点小推广
创作不易,希望可以支持下我的开源软件,及我的小工具,欢迎来 gitee 点星,fork ,提 bug 。
Excel 通用导入导出,支持 Excel 公式
博客地址:https://blog.csdn.net/sanri1993/articlehttps://img.qb5200.com/download-x/details/100601578
gitee:https://gitee.com/sanri/sanri-excel-poi
使用模板代码 ,从数据库生成代码 ,及一些项目中经常可以用到的小工具
博客地址:https://blog.csdn.net/sanri1993/articlehttps://img.qb5200.com/download-x/details/98664034
gitee:https://gitee.com/sanri/sanri-tools-maven
加载全部内容