【Spring源码】22. 属性填充populateBean()详解

【Spring源码系列- IOC】

1

【Spring源码】0.安装Gradle环境

2

【Spring源码】1.下载与编译_pom relocation to an other version number is not f

3

【Spring源码】2.试个水先~Debug找到传说中的三级缓存(图解向,堆图预警)

4

【Spring源码】3. xml文件如何转换成BeanDefinition(主要涉及prepareRefresh()+ obtainFreshBeanFactory()两个函数,图解向,堆图预警)_spring xml转bean

5

【Spring源码】4. 自己搞个标签?~自定义标签保姆级全过程(图解向,堆图预警)

6

【Spring源码】5.spring的bean工厂准备工作(prepareBeanFactory(beanFactory)

7

【Spring源码】6. Spring扩展自定义属性编辑器保姆级教程

8

【Spring源码】7. 如何添加自定义的BeanFactoryPostProcessor

9

【Spring源码】8. 捋下invokeBeanFactoryPostProcessors()主要处理流程

10

【Spring源码】9. 超级重要的ConfigurationClassPostProcessor

11

【Spring源码】10. 递归调用的processConfigurationClass()方法

12

【Spring源码】11. 我是注解类不?checkConfigurationClassCandidate()注解类判断方法详解

13

【Spring源码】12. 注册bean处理器registerBeanPostProcessors()

14

【Spring源码】13. 国际化处理initMessageSource()源码解析

【补充内容】【保姆级】SpringBoot项目中的i18n国际化

15

【Spring源码】14. 消息多播器(观察者模式)

【补充内容】【保姆级示例向】观察者模式

16

【Spring源码】15. Bean的创建过程(1.概述篇)

17

【Spring源码】16. Bean的创建过程(2)

18

【Spring源码】17.创建Bean这篇认真的@(・●・)@

【补充内容】

【保姆级·创建对象】如何通过Supplier创建对象

【保姆级·创建对象】如何通过factory-method创建对象

【保姆级·创建对象】如何利用resolveBeforeInstantiation()在预处理阶段返回一个Bean的实例对象

19

【Spring源码】18. factory-method创建对象关键函数详解:instantiateUsingFactoryMethod()

20

【Spring源码】19. 没合适的构造器?找determineCandidateConstructors()!

21

【Spring源码】20. MergedBeanDefinitionPostProcessor修改/合并bean定义

【补充内容】

【保姆级】@PostConstruct & @PreDestroy使用示例

【Spring源码】AutowiredAnnotationBeanPostProcessor.postProcessMergedBeanDefinition()详解

【Spring源码】CommonAnnotationBeanPostProcessor.postProcessMergedBeanDefinition()详解

22

【Spring源码】21. 初探循环依赖

【补充内容】

【保姆级】手把手Debug循环依赖的整体流程

【实践向】当移除了三级缓存……

【分析向】没有三级缓存会导致什么?

【Spring源码】插播一个创建代理对象的wrapIfNecessary()方法

23

【Spring源码】22. 属性填充populateBean()详解

【补充内容】

【Spring源码】自动注入·名称:autowireByName()详解

【Spring源码】自动注入·类型:autowireByType()详解

【Spring源码】属性值的解析与赋值:populateBean().applyPropertyValues()

【保姆级】超超超简单的自定义注解实现@Autowired同款功能

24

【Spring源码】23. 执行初始化逻辑:initializeBean()

进入populateBean()

对bean的属性进行填充,将各个属性值注入(存在其他bean的属性,则会递归初始化依赖的bean)

一开始会先对传入的参数进行判断(如下图红框框中的逻辑)

  • 如果传入的BeanWrapper和RootBeanDefinition的propertyValues属性都为空,则表示无可填充的属性直接结束方法;

  • 如果传入的BeanWrapper为空但是RootBeanDefinition的propertyValues属性不为空,则会抛出BeanCreationException(mbd.getResourceDescription(), beanName, "Cannot apply property values to null instance")的异常;

  • 只有在传入的BeanWrapper不为空才会进入到后面的逻辑

一般只要经历了前面的创建过程,这个位置传入的BeanWrapper就不会为空

有咩有客官觉得if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors())这个判断很熟悉( ̄∇ ̄)

前面介绍过的resolveBeforeInstantiation()这个方法中就有类似的代码(这个方法是应用实例化前的前置处理器,给BeanPostProcessors一个机会来返回代理代替真正的实例)

对resolveBeforeInstantiation()这个方法感兴趣的客官欢迎移步

【保姆级·创建对象】如何利用resolveBeforeInstantiation()在预处理阶段返回一个Bean的实例对象

回到我们的populateBean()

支持拓展的赋值方式(postProcessAfterInstantiation())

这部分给任何实现了InstantiationAwareBeanPostProcessors的子类机会去修改bean的状态,在设置属性之前,可以被用来支持类型的字段注入,这部分包含两个判断:

  • !mbd.isSynthetic()是对属性synthetic 进行的判断,一般只有AOP相关的pointCut配置或者Advice配置才会将synthetic(合成的)设置为true

  • hasInstantiationAwareBeanPostProcessors()是判断是否有实现了InstantiationAwareBeanPostProcessors的子类

同时满足两个条件(传入的mbd不是synthetic有实现了InstantiationAwareBeanPostProcessors的子类)会执行ibp.postProcessAfterInstantiation()方法

这个方法默认返回true(它有几个实现类,感兴趣的可以都点进去看下,都是返回的true,当然你也可以自定义一个类来实现InstantiationAwareBeanPostProcessor返回个false直接终止后续的赋值操作)

接着获取beanDefinition中property属性(需要手动定义,默认是🈳,PropertyValues是包含一个或者多个PropertyValue对象的容器)

判断自动装配模式(autowireByName()/autowireByType())

先获取mbd的自动装配模式,然后再判断mbd的自动装配模式(这里判断了两种模式):

  • AUTOWIRE_BY_NAME 根据名称自动注入

  • AUTOWIRE_BY_TYPE 根据类型自动注入

还有其他的,具体如下图

想详细了解autowireByName()和autowireByType()这两个方法的客官可以蹲下,后面会详细介绍

此时newPvs已经包含了pvs的属性值以及通过AUTOWIRE_BY_NAME/AUTOWIRE_BY_TYPE两种自动装配模式所得到的属性值(其实就是根据自动装配模式进行了分类,后面才会进行注入),将其赋值给pvs

完成对象/属性的注入(postProcessProperties())

接下来定义了hasInstAwareBpps和needsDepCheck这两个布尔值分别用于判断工厂是否拥有InstantiationAwareBeanPostProcessor后处理器,以及判断是否需要依赖检查(默认AbstractBeanDefinition.DEPENDENCY_CHECK_NONE不检查),后面会分别对这两个值进行判断

  • 当hasInstAwareBpps为true时,即工厂有InstantiationAwareBeanPostProcessor,那么就遍历工厂所有的后置处理器

解析对象中定义的@Autowired注解,并通过调用这些后置处理器的postProcessProperties()方法,完成对象/属性的注入

当needsDepCheck为true时,会进行行依赖检查,判断filteredPds是否为空,如果为空会对其进行赋值,最终通过调用checkDependencies()检查pd的setter()方法需要赋值时,pvs中有没有满足pd需求的属性值可供其赋值

属性值的解析和赋值(applyPropertyValues())

通过property标签定义的属性值完成各种属性值的解析和赋值

至此方法populateBean()执行结束,撒个花吧嘿嘿(。・ω・。)ノ🎉

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

AQin1012

求小鱼干呢~~

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值