一篇文章帶你玩轉(zhuǎn)Spring bean的終極利器
前言
前面的篇幅里有提到通過(guò)InitializingBean和Disposable等接口可以對(duì)bean的初始化和銷毀做一些自定義操作,那么有一點(diǎn)要注意,那僅僅是在bean被容器實(shí)例化之后的操作,在spring的世界里,要想對(duì)實(shí)例化這個(gè)過(guò)程做點(diǎn)什么,作為一個(gè)普通業(yè)務(wù)的開發(fā)人員,顯然不需要去繼承ApplicationContext或者BeanFactory,因?yàn)閟pring container為我們提供了一些接口,讓我們以插件的形式去擴(kuò)展BeanFactory對(duì)bean的初始化操作,其中就有我們今天的主角——BeanPostProcessor(以下簡(jiǎn)稱bpp)接口。
源碼,先睹為快
這個(gè)用法很簡(jiǎn)單,它只有兩個(gè)方法,我們實(shí)現(xiàn)自己的BeanPostProcessor,Spring能自動(dòng)注冊(cè)到容器中。
其中before方法是在bean實(shí)例化之后,屬性設(shè)置之后但在初始化方法之前執(zhí)行;after方法是在各種初始化方法之后執(zhí)行。
說(shuō)到這里可能有人會(huì)想,這跟生命周期中的其它初始化接口有啥區(qū)別?其它的初始化方法也可以修改bean啊,這個(gè)問(wèn)題問(wèn)得好,那么我們來(lái)說(shuō)下這個(gè)接口與InitializingBean Disposable接口以及自定義的init destroy方法的本質(zhì)區(qū)別
- bpp的兩個(gè)方法是有返回值Object的,這恰恰是問(wèn)題的關(guān)鍵,這個(gè)bean就是我們要修改的bean,這樣一來(lái),我們就可以修改bean實(shí)例本身,或替換,或wrap成一個(gè)proxy bean(Spring中的aop機(jī)制多是這么干),而其它的初始化接口的返回為void,因此它們理論上只能修改bean的狀態(tài),能做的東西相當(dāng)受限制。
- bpp是以擴(kuò)展插件的形式被Container執(zhí)行,不需要bean本身去做什么(bean本身不用實(shí)現(xiàn)這個(gè)接口),所以這個(gè)插件跟bean在代碼上不耦合
- 在執(zhí)行方式上也完全不同,bpp是作為Spring container的一個(gè)擴(kuò)展,在容器初始化bean的過(guò)程過(guò),對(duì)每個(gè)bean都會(huì)執(zhí)行一次,而初始化接口,由于是特定的bean實(shí)現(xiàn)的,所以與其它的bean無(wú)關(guān),只對(duì)初始該類型的bean執(zhí)行。簡(jiǎn)而言之就是,雖然都是由容器執(zhí)行對(duì)bean的操作,bpp是擴(kuò)展的容器本身行為,而初始化接口是擴(kuò)展bean的行為后被容器執(zhí)行的。
在這里有兩種特殊的bpp不得不說(shuō),假設(shè)你需要自定義一個(gè)類似于@Autowire或者@Inject的注入功能的注解的時(shí)候(你可能會(huì)用到InjectionMetadata),普通的bpp可能就滿足不了你的需要了,你可能用到兩個(gè)特殊的bpp。
MergedBeanDefinitionPostProcessor(以下簡(jiǎn)稱mbdpp)
InstantiationAwareBeanPostProcessor(以下簡(jiǎn)稱iabpp)
他們都是繼承自bpp,但在spring bean 創(chuàng)建的過(guò)程中切入點(diǎn)不同于普通的bpp。
InstantiationAwareBeanPostProcessor接口
看注釋
postProcessBeforeInstantiation方法
查閱AbstractAutowireCapableBeanFactory的createBean方法(這個(gè)方法是Spring容器創(chuàng)建bean的核心方法),可以看到,postProcessBeforeInstantiation是在bean實(shí)例化之前,postProcessAfterInstantiation是在實(shí)例化之后屬性設(shè)置以及autowire注入之前,它一般是spring框架內(nèi)部使用,但在這里大有可為,用postProcessBeforeInstantiation可以生成代理對(duì)象( 一般作法是讓postProcessorBeforeInstantiation方法返回不為null,這樣就會(huì)中斷后續(xù)創(chuàng)建bean實(shí)例的過(guò)程,會(huì)以這個(gè)方法返回的對(duì)象作為bean實(shí)例),看源碼:
postProcessPropertyValues方法
用postProcessPropertyValues 可以完成對(duì)屬性的各種操作,注解中元數(shù)據(jù)的解析等,Spring的@Autowire注入,JSR330的@Inject以及JSR250的@Resource等注入操作都是通過(guò)這個(gè)方法完成。
這接口的用處在spring底層較多,有興趣的同學(xué)可以翻閱源碼,以下是兩個(gè)比較典型的實(shí)現(xiàn)。
AutowiredAnnotationBeanPostProcessor
AbstractAutoProxyCreator
MergedBeanDefinitionPostProcessor接口
這個(gè)接口傳入了一個(gè)RootBeanDefinition,這里允許我們修改bean的定義,@AutuwiredAnnotationBeanPostProcessor通過(guò)實(shí)現(xiàn)這個(gè)方法檢查并注冊(cè)需要注入的成員。
BeanFactoryPostProcessor(bfpp)
除了BeanPostProcessor還有一種想必大家都知道,那就是BeanFactoryPostProcessor
bfpp是作為beanFactory的一個(gè)很重要擴(kuò)展插件,可以用來(lái)自定義BeanDefination的。它與bpp主要區(qū)別在于:
- bpp是處理的bean實(shí)例,bfpp是處理bean的定義
- bfpp能讀取和修改bean的定義(BeanDefination),比如說(shuō)在配置中屬性值用到的占位符${}就是PropertyPlaceholderConfigurer通過(guò)實(shí)現(xiàn)bfpp來(lái)實(shí)現(xiàn)的
- bpp處理的則是新鮮出爐并且設(shè)置好屬性的bean的實(shí)例(上邊提到iabpp和mdbpp的兩種特殊的bpp除外)
好了,說(shuō)了這么多,來(lái)看下Spring創(chuàng)建bean的大致流程圖,這里只標(biāo)出了比較關(guān)鍵的節(jié)點(diǎn)
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
java根據(jù)模板導(dǎo)出PDF的詳細(xì)實(shí)現(xiàn)過(guò)程
前段時(shí)間因?yàn)橄嚓P(guān)業(yè)務(wù)需求需要后臺(tái)生成pdf文件,所以下面這篇文章主要給大家介紹了關(guān)于java根據(jù)模板導(dǎo)出PDF的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-02-02使用java編程從0到1實(shí)現(xiàn)一個(gè)簡(jiǎn)單計(jì)算器
這篇文章主要介紹了使用java編程從0到1實(shí)現(xiàn)一個(gè)簡(jiǎn)單計(jì)算器,文章中用代碼實(shí)例講解的很清晰,有感興趣的同學(xué)可以學(xué)習(xí)研究下2021-02-02關(guān)于@Autowired的使用及注意事項(xiàng)
這篇文章主要介紹了關(guān)于@Autowired的使用及注意事項(xiàng),具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-05-05java實(shí)現(xiàn)兩臺(tái)服務(wù)器間文件復(fù)制的方法
這篇文章主要介紹了java實(shí)現(xiàn)兩臺(tái)服務(wù)器間文件復(fù)制的方法,是對(duì)單臺(tái)服務(wù)器上文件復(fù)制功能的升級(jí)與改進(jìn),具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-01-01Java線程之間數(shù)據(jù)傳遞的實(shí)現(xiàn)示例(4種)
我們經(jīng)常會(huì)遇到父子線程數(shù)據(jù)傳遞(非調(diào)用參數(shù))的場(chǎng)景,本文主要介紹了Java線程之間數(shù)據(jù)傳遞的實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-08-08在service層注入mapper時(shí)報(bào)空指針的解決
這篇文章主要介紹了在service層注入mapper時(shí)報(bào)空指針的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06