本文主要讲解Spring生命周期中实例化后的方法。方法入口为populateBean(beanName, mbd, instanceWrapper)
,该方法除了调用InstantiationAwareBeanPostProcessor接口的postProcessAfterInstantiation方法(实例化后扩展接口),还会进行属性的依赖注入,即给Bean中符合要求的字段进行属性填充。
文章目录
- 一、DI整体流程图
- 二、需明白知识点
- 三、待补充知识点
一、DI整体流程图
我将具体的依赖注入步骤整理为如下的流程图:
不同版本的Spring该部分代码有些许差异,但整体思路差不多,我的版本是Spring5.0.x
网页端是能够清晰且方便展示的,手机端可以保存后图库查看
自认为该流程图包含了70%以上的依赖注入的内容,如果你正在阅读依赖注入部分源码,可以参考上面的流程
二、需明白知识点
我相信你在跟完依赖注入的源代码后,你将要能够明白以下问题:
- @Autwired和@Resource在注入Bean时的name和type优先级区别
- @Autwired注解在解析方法或者成员变量的解析详细流程(三个CandidateResolver类、两个注解、一个byName)
- 参数和方法是static类型时能否被注入
- @Bean中配置AutworedMode类型为name或type后和默认情况什么区别
- 当我们在成员变量和set方法上(成员变量对应的set方法)都是用了@Autwired注解后,两个@Autwired的注入顺序是什么
- @Value注解解析的具体规则
- 如果我们自定义了对应的属性注入参数(代码如下),代码中对应的逻辑判断在什么位置。提示:如果我们自定义了相关Bean的属性注入内容,那么在注解解析的时候,会跳过我们定义过的属性。
// 如果bean的名字是teacher,那么久给teacher对象的name字段赋值为mobian
@Component
public class MergedBeanPostProcess implements MergedBeanDefinitionPostProcessor {
@Override
public void postProcessMergedBeanDefinition(RootBeanDefinition beanDefinition, Class<?> beanType, String beanName) {
if(beanName.equals("teacher")){
beanDefinition.setInitMethodName("a");
beanDefinition.getPropertyValues().add("name","mobian");
}
}
}
三、待补充知识点
图中还有几个重要知识点没有详细展开,需要你自行研究。
1、注入的Bean存入缓存部分
当我们给对应的方法或者是成员变量属性注入成功后,会将该Bean
构建一个ShortcutDependencyDescriptor类的对象作为缓存,保存了当前成员变量或者方法所匹配的autowiredBeanName,后期如果不同Bean对象使用到了同一个注入的Bean,就可以直接去缓存中,根据beanName直接去单例池中获取
2、注入的类型是FactoryBean类型
当我们的注入类型是FactoryBean的时候,它会在逻辑校验中调用到对应的getObjectType。我们之前使用实现FactoryBean接口的方式定义Bean,都需要重写getObject和getObjectType方法,前者在我们获取Bean的时候会被调用,后者在此处我们去获取Bean的类型的时候就会使用到,这里我们可以不需要去实例化对象就能获取到该Bean的类型
3、对于属性描述器部分
对于一个标准的pojo类,都是一个成员变量,然后配置上对应的get、set方法。每一个成员变量都可以映射为一个PropertyDescriptor类的对象,该类包含对应成员变量的名字、set方法、get方法…这个属性描述器贯穿了整个依赖注入的源码。