Spring Aware 到底是什么?


通過如下前序兩篇文章:

  1. Spring Bean 生命周期之“我從哪里來”?
  2. Spring Bean 生命周期之“我要到哪里去”?
    我們了解了 Spring Bean 的生命周期核心內容,bean 是如何被初始化變為 Ready for Use 的狀態,當資源被回收時又是如何被 destroy 的,但 Spring Bean Life Cycle圖並未被全部點亮,這篇文章將點亮剩余內容,同時說說你常見的 XxxxAware 接口

為什么要說 Spring Bean 生命周期又說 Aware 呢?下來點亮剩下內容你也許就明白了:

  1. 在 Spring Bean Ready for Use之前的起源當然是要調用構造器,所以 Constructor 毋庸置疑是創建 Spring Bean 的第一步
  2. 通過 Setter 方法完成依賴注入,SDI (Setter Dependency Injection)
  3. 依賴注入一旦結束,BeanNameAware.setBeanName() 會被調用,它設置該 bean 在 Bean Factory 中的名稱
  4. 接下來調用 BeanClassLoaderAware.setBeanClassLoader() ,為 bean 實例提供類加載器,我們知道所有類都是要通過類加載器加載到上下文的,關於類的加載機制/雙親委派模型(大廠都愛問的面試題)內容會在后續給出來,讓你透徹的了解
  5. 然后 BeanFactoryAware.setBeanFactory() 會被調用為 bean 實例提供其所擁有的 factory

關於 1、2 兩點我要額外多說一些內容,請看下面代碼:

這里,我們嘗試通過構造器訪問自動注入的 field Environment env,當構造器被調用時,Spring Bean 還沒被完全初始化,這就會導致 NullPointerExceptions
我們變換一下方式:

這種方式,Environment 實例被安全注入之后才調用 @PostConstruct標記的方法,這樣就不會拋出 NullPointerException 了。

這會回看周期圖,有沒有豁然開朗?

敲黑板

等所有 Spring Bean 都完成依賴注入(周期圖中的 Setter Methods 部分)再使用 bean 的引用才是安全的方式,

后續會有一個章節專門說一說面試經常被問起的 Spring 有幾種依賴注入方式的尷尬問題,請關注后續文章

到這里終於可以說一說 Aware 了,且看

Aware

Aware 翻譯過來可以理解為"察覺的;注意到的;感知的" ,XxxxAware 也就是對....感知的,沒有 Aware 就是無感知的嗎?對嘍

Spring 的依賴注入最大亮點就是所有的 Bean 對 Spring 容器的存在是沒有意識的,拿 Spring Bean 生命周期之“我從哪里來”? 文章中“小學生入少先隊”為例子說明,小學生還是那個小學生,加入少先隊還是加入共青團只不過規則不一樣罷了
但是在實際項目中,我們不可避免的要用到 Spring 容器本身提供的資源(難免要有事情需要少先隊組織的幫助),這時候要讓 Bean 主動意識到 Spring 容器的存在,才能調用 Spring 所提供的資源,這就是 Spring Aware. 其實 Spring Aware 是 Spring 設計為框架內部使用的,若使用了,你的 Bean 將會和 Spring 框架耦合,所以自己不單獨使用,但是在讀框架源碼時希望你不再模糊.

常見的 Spring Aware 接口

Aware子接口 描述
BeanNameAware 獲取容器中 Bean 的名稱
BeanFactoryAware 獲取當前 BeanFactory ,這樣可以調用容器的服務
ApplicationContextAware 同上,在BeanFactory 和 ApplicationContext 的區別 中已明確說明
MessageSourceAware 獲取 Message Source 相關文本信息
ApplicationEventPublisherAware 發布事件
ResourceLoaderAware 獲取資源加載器,這樣獲取外部資源文件

來看類關系圖:

當然不止以上這些 Aware, 通常使用 Spring Aware 的目的是為了讓 Bean 獲得 Spring 容器的服務。

代碼示例

BeanNameAware

自定義 bean 實現 BeanNameAware

注冊 bean

運行

和預想一樣,Bean Name 輸出結果為 myCustomBeanName,如果移除掉 @Bean 注解的 name 屬性, 輸出結果為 getMyBeanName

總結

在大多數情況下,我們應該避免使用任何 Aware 接口,除非我們需要它們。實現這些接口會將代碼耦合到Spring框架,但是希望看過本節內容之后閱讀框架源碼思維更加清晰

靈魂追問

  1. 框架中有哪些經典的 Aware 應用?
  2. 到現在你能很好的理解 Spring Bean 的生命周期嗎?

Demo代碼

涉及到 Spring Bean 生命周期的測試代碼由於內容較多,沒有寫在此處,關注公眾號並回復 「demo」獲取相關代碼,請自行嘗試運行結果


提高效率工具


歡迎持續關注公眾號:「日拱一兵」

  • 前沿 Java 技術干貨分享
  • 高效工具匯總
  • 面試問題分析與解答
  • 技術資料領取

后續會出一系列文章點亮上圖,同時進行 Spring 知識點解釋與串聯,在工作中充分利用 Spring 的特性
另外,還會推出 Java 多線程與 ElasticSearch 相關內容

持續關注,帶你像讀偵探小說一樣輕松趣味學習 Java 技術棧相關知識


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM