spring实现先返回后执行_Spring后置处理器大盘点

在前面几篇文章中梳理了Spring中bean的创建过程,在这个过程中各式各样的后置处理器发挥了不同的作用,可以说后置处理器贯穿了bean的实例化以及初始化过程。在这篇文章中,将按照出场顺序对后置处理器作用场景及发挥功能进行梳理。

调用1

InstantiationAwareBeanPostProcessor

postProcessBeforeInstantiation()

 AbstractAutowireCapableBeanFactory 的createBean方法中调用,这时bean还没有被实例化:

581644b46ce8bd9cc548481098e21cb4.png

调用resolveBeforeInstantiation方法:

6fd43fb22d848dce884788d32c625f84.png

applyBeanPostProcessorsBeforeInstantiation方法:

37f48c83a9a0885da5aea60fda7b192a.png

在这里,首先拿到spring中提供的所有后置处理器,判断是不是InstantiationAwareBeanPostProcessor。该后置处理器实现了BeanPostProcessor,在这调用了postProcessBeforeInstantiation方法。

这里在目标对象被spring实例化之前调用,postProcessBeforeInstantiation方法的返回值类型是Object,可以返回任何类型的值。由于此时目标对象还未实例化,所以这个返回值可以用来代替原本该生成的目标对象的实例,一般为代理对象。

如果该方法的返回的Object对象代替了原本该生成的目标对象,那么就会把返回的对象放到单例池当中缓存,后续只有BeanPostProcessor的postProcessAfterInitialization方法会调用,其它方法不再调用。

如果这里返回了null,就按照正常的流程创建对象,交给spring去负责对象的实例化。因此这个方法可以判断这个对象在spring实例化之前是否要做特殊的处理,比如不交给Spring管理,自己使用代理产生。

调用2

SmartInstantiationAwareBeanPostProcessor

determineCandidateConstructors()

在AbstractAutowireCapableBeanFactory 的 createBeanInstance方法中调用:

90bc46364f77dd0f2fd0fd4c8adc815f.png

determineConstructorsFromBeanPostProcessors方法,该方法用于推断实例化的构造方法,这里可能检测出bean拥有多个候选构造方法:

96fccf6266a41b29f3be2874ffc63ccc.png

SmartInstantiationAwareBeanPostProcessor接口的实现类AutowiredAnnotationBeanPostProcessor负责完成这个过程,如果有多个构造方法的情况下,ctors会返回空,后续使用默认无参构造方法进行实例化。但是如果有一个构造方法上有@Autowired注解,spring会优先选择这个方法。

调用3

MergedBeanDefinitionPostProcessor

postProcessMergedBeanDefinition()

AbstractAutowireCapableBeanFactory的doCreateBean 方法中调用:

2e2bde126be51c7feae45f6e106915e2.png

applyMergedBeanDefinitionPostProcessors方法:

2661436ef824c5ab0ce90ece0c1a60e9.png

在方法中对所有实现了MergedBeanDefinitionPostProcessor接口的后置处理器进行遍历,这里具体调用AutowiredAnnotationBeanPostProcessor,用于扫描需要注入的属性。

AutowiredAnnotationBeanPostProcessor中,定义了两种需要扫描的注解类型,@Autowired和@Value

7032bad46f49a45f9a790a66ff279db3.png

在findAutowiredAnnotation方法中:

7b264c2cf4c4f969c188143c48530abd.png

对正在创建的bean进行扫描,如果有属性和方法上面加了这两个注解,就会把对应的方法或者属性保存,最终在buildAutowiringMetadata方法中封装成InjectionMetadata对象。

0e6153856c9f8038f7ea1ff638f7b728.png

需要注意这里的后置处理器仅仅用于扫描及缓存bean的注入信息,这里只完成了查找功能,没有完成属性的注入,属性的注入是在之后的另外的后置处理器中完成的。

调用4

SmartInstantiationAwareBeanPostProcessor

getEarlyBeanReference()

在AbstractAutowireCapableBeanFactory的doCreateBean 方法中调用,主要用于处理Bean的循环依赖:

3b1e6be6e233b717e2d6b28ae9122a7c.png

在产生循环依赖后调用getEarlyBeanReference方法:

d274235c08aa899db3a18a5c37666e83.png

在这里遍历后置处理器,得到经过后置处理器代理后的对象,放入spring的二级缓存当中,提前暴露供循环引用的情况调用。注意这里返回的仅仅是一个对象,还算不上是一个完整的bean对象。这个具体调用过程在上一篇讲循环依赖的中的文章中讲的比较详细,如果有不明白的可以回顾一下。

调用5

InstantiationAwareBeanPostProcessor

postProcessAfterInstantiation()

在AbstractAutowireCapableBeanFactory的populateBean方法中调用:

a9c1accba3ff9ec69cc43b9ac6a01772.png

在populateBean方法中:

5bf0425752aa437a9da9d56535092b70.png

该方法在目标对象实例化之后调用,这个时候对象已经被实例化,但是该实例的属性还未被设置,都是null。

这里遍历后置处理器,如果实现了InstantiationAwareBeanPostProcessor,那么就调用postProcessAfterInstantiation方法。如果方法返回值为true,按照正常流程进行属性值的填充;如果该方法返回false,会忽略属性值的设置过程。简而言之,该后置处理器用于判断当前实例化完成的bean需不需要进行属性填充。

调用6

InstantiationAwareBeanPostProcessor

postProcessPropertyValues()

同样在populateBean方法中,在postProcessAfterInstantiation后返回true时执行,如果方法返回false,该方法不会被调用。

dceea7ce4a6b8ce64cc21bbfeb0d11e9.png

遍历后置处理器,如果属于InstantiationAwareBeanPostProcessor类型,则调用它的postProcessPropertyValues方法。

这里发挥作用的是AutowiredAnnotationBeanPostProcessor,负责对添加了 @Autowired和@Value等注解的属性进行依赖的填充。在其中遍历所有需要注入的属性的列表,使用反射将注入的bean实例赋值给属性。(具体过程参照Spring实例化Bean源码解析)

调用7

BeanPostProcessor

postProcessBeforeInitialization()

AbstractAutowireCapableBeanFactory的doCreateBean方法中调用initializeBeanfan方法:

d6e271c88cb4324a03eb07ce5823bde3.png

applyBeanPostProcessorsBeforeInitialization方法中:

5f0563c5ec09674886cf74f53101f5e9.png

在该方法中,遍历执行所有BeanPostProcessor的postProcessBeforeInitialization方法。

d52296325c5d9113aa645c4e10aa7318.png

在执行该方法前,bean已经被实例化完成,并且完成了属性的填充,因此这个过程属于后续的bean的初始化过程。

需要注意的是,如果在bean中有方法被标注了@PostContrust注解,那么在CommonAnnotationBeanPostProcessor中,会调用该@PostContrust方法。

调用8

BeanPostProcessor

postProcessAfterInitialization()

和第7次调用入口相同,也是在AbstractAutowireCapableBeanFactory 的initializeBean方法中:

40734154819bbdb49e9decf164431f3e.png

applyBeanPostProcessorsAfterInitialization方法中:

571320a7437d254fd5e50cb3f5f893e5.png

遍历执行所有BeanPostProcessor的postProcessAfterInitialization方法。综上所述,bean的各种方法执行属性为,先执行构造方法,再执行后置管理器中的before方法及@PostContrust方法,最后执行后置处理器的after方法。

调用9

InitDestroyAnnotationBeanPostProcessor

postProcessBeforeDestruction()

如果当前bean中有方法被@PreDestroy注解标注,那么当Spring的ApplicationContext执行close方法时调用该后置处理器。在DefaultSingletonBeanRegistry中执行destroyBean方法:

d2dd2d29770c3e7e35bed40a77f006a7.png

调用destroy方法:

304aa357e2d3717f07390953a0f3bb4e.png

InitDestroyAnnotationBeanPostProcessor的postProcessBeforeDestruction方法:

029fa4fdf057899a346bfb4446790dcd.png

在该方法中,调用@PreDestroy注解标注的方法,执行销毁方法。

总结

本文对贯穿bean的实例化及初始化过程中出现的后置处理器进行了一下梳理,但是还有很多其他的后置处理器没有讲到。可以说后置处理器是spring提供给使用者的一些扩展点,如果能够熟练的使用这些后置处理器,能够帮助我们接触到一些spring中比较深层的东西,并对spring中的生命周期进行有利的插手。

公众号后台回复

"面试"---领取大厂面试资料

"导图"---领取24张Java后端学习笔记导图

"架构"---领取29本java架构师电子书籍

"实战"---领取springboot实战项目

"视频"---领取最新java架构师视频

"加群"---加入学习交流群

1c17027067818be0f8cb827172c7d468.png

扫码关注公众号

有趣、深入、直接

与你聊聊java

觉得有用,点个在看吧~
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值