一篇文章帶你玩轉(zhuǎn)Spring bean的終極利器
前言
前面的篇幅里有提到通過InitializingBean和Disposable等接口可以對bean的初始化和銷毀做一些自定義操作,那么有一點要注意,那僅僅是在bean被容器實例化之后的操作,在spring的世界里,要想對實例化這個過程做點什么,作為一個普通業(yè)務(wù)的開發(fā)人員,顯然不需要去繼承ApplicationContext或者BeanFactory,因為spring container為我們提供了一些接口,讓我們以插件的形式去擴展BeanFactory對bean的初始化操作,其中就有我們今天的主角——BeanPostProcessor(以下簡稱bpp)接口。
源碼,先睹為快

這個用法很簡單,它只有兩個方法,我們實現(xiàn)自己的BeanPostProcessor,Spring能自動注冊到容器中。
其中before方法是在bean實例化之后,屬性設(shè)置之后但在初始化方法之前執(zhí)行;after方法是在各種初始化方法之后執(zhí)行。
說到這里可能有人會想,這跟生命周期中的其它初始化接口有啥區(qū)別?其它的初始化方法也可以修改bean啊,這個問題問得好,那么我們來說下這個接口與InitializingBean Disposable接口以及自定義的init destroy方法的本質(zhì)區(qū)別
- bpp的兩個方法是有返回值Object的,這恰恰是問題的關(guān)鍵,這個bean就是我們要修改的bean,這樣一來,我們就可以修改bean實例本身,或替換,或wrap成一個proxy bean(Spring中的aop機制多是這么干),而其它的初始化接口的返回為void,因此它們理論上只能修改bean的狀態(tài),能做的東西相當受限制。
- bpp是以擴展插件的形式被Container執(zhí)行,不需要bean本身去做什么(bean本身不用實現(xiàn)這個接口),所以這個插件跟bean在代碼上不耦合
- 在執(zhí)行方式上也完全不同,bpp是作為Spring container的一個擴展,在容器初始化bean的過程過,對每個bean都會執(zhí)行一次,而初始化接口,由于是特定的bean實現(xiàn)的,所以與其它的bean無關(guān),只對初始該類型的bean執(zhí)行。簡而言之就是,雖然都是由容器執(zhí)行對bean的操作,bpp是擴展的容器本身行為,而初始化接口是擴展bean的行為后被容器執(zhí)行的。
在這里有兩種特殊的bpp不得不說,假設(shè)你需要自定義一個類似于@Autowire或者@Inject的注入功能的注解的時候(你可能會用到InjectionMetadata),普通的bpp可能就滿足不了你的需要了,你可能用到兩個特殊的bpp。
MergedBeanDefinitionPostProcessor(以下簡稱mbdpp)
InstantiationAwareBeanPostProcessor(以下簡稱iabpp)
他們都是繼承自bpp,但在spring bean 創(chuàng)建的過程中切入點不同于普通的bpp。
InstantiationAwareBeanPostProcessor接口

看注釋

postProcessBeforeInstantiation方法
查閱AbstractAutowireCapableBeanFactory的createBean方法(這個方法是Spring容器創(chuàng)建bean的核心方法),可以看到,postProcessBeforeInstantiation是在bean實例化之前,postProcessAfterInstantiation是在實例化之后屬性設(shè)置以及autowire注入之前,它一般是spring框架內(nèi)部使用,但在這里大有可為,用postProcessBeforeInstantiation可以生成代理對象( 一般作法是讓postProcessorBeforeInstantiation方法返回不為null,這樣就會中斷后續(xù)創(chuàng)建bean實例的過程,會以這個方法返回的對象作為bean實例),看源碼:

postProcessPropertyValues方法
用postProcessPropertyValues 可以完成對屬性的各種操作,注解中元數(shù)據(jù)的解析等,Spring的@Autowire注入,JSR330的@Inject以及JSR250的@Resource等注入操作都是通過這個方法完成。
這接口的用處在spring底層較多,有興趣的同學可以翻閱源碼,以下是兩個比較典型的實現(xiàn)。
AutowiredAnnotationBeanPostProcessor
AbstractAutoProxyCreator
MergedBeanDefinitionPostProcessor接口

這個接口傳入了一個RootBeanDefinition,這里允許我們修改bean的定義,@AutuwiredAnnotationBeanPostProcessor通過實現(xiàn)這個方法檢查并注冊需要注入的成員。
BeanFactoryPostProcessor(bfpp)
除了BeanPostProcessor還有一種想必大家都知道,那就是BeanFactoryPostProcessor
bfpp是作為beanFactory的一個很重要擴展插件,可以用來自定義BeanDefination的。它與bpp主要區(qū)別在于:
- bpp是處理的bean實例,bfpp是處理bean的定義
- bfpp能讀取和修改bean的定義(BeanDefination),比如說在配置中屬性值用到的占位符${}就是PropertyPlaceholderConfigurer通過實現(xiàn)bfpp來實現(xiàn)的
- bpp處理的則是新鮮出爐并且設(shè)置好屬性的bean的實例(上邊提到iabpp和mdbpp的兩種特殊的bpp除外)
好了,說了這么多,來看下Spring創(chuàng)建bean的大致流程圖,這里只標出了比較關(guān)鍵的節(jié)點

總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
Java算法實現(xiàn)調(diào)整數(shù)組順序使奇數(shù)位于偶數(shù)之前的講解
今天小編就為大家分享一篇關(guān)于Java算法實現(xiàn)調(diào)整數(shù)組順序使奇數(shù)位于偶數(shù)之前的講解,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-01-01
Spring Cloud Feign內(nèi)部實現(xiàn)代碼細節(jié)
Feign 的英文表意為“假裝,偽裝,變形”, 是一個http請求調(diào)用的輕量級框架,可以以Java接口注解的方式調(diào)用Http請求,而不用像Java中通過封裝HTTP請求報文的方式直接調(diào)用。接下來通過本文給大家分享Spring Cloud Feign內(nèi)部實現(xiàn)代碼細節(jié),感興趣的朋友一起看看吧2021-05-05
Java JDBC連接Kerberos認證的HIVE和Impala方式
本文主要介紹了HiveJDBC和ImpalaJDBC的使用方法,包括版本對應(yīng)、Maven安裝、主機名配置、端口開通、JDBC連接和Kerberos認證等2025-02-02
Java中Queue的poll()和remove()區(qū)別詳解
這篇文章主要介紹了Java中Queue的poll()和remove()區(qū)別詳解,Queue接口提供了許多方法,其中poll()和remove()是兩個常用的方法,它們的區(qū)別在于,當隊列為空時,poll()方法返回null,而remove()方法會拋出,需要的朋友可以參考下2023-07-07

