深究Spring中Bean的生命周期

前言

這其實是一道面試題直秆,是我在面試百度的時候被問到的,當時沒有答出來(因為自己真的很菜)萨咳,后來在網上尋找答案培他,看到也是一頭霧水舀凛,直到看到了《Spring in action》這本書猛遍,書上有對Bean聲明周期的大致解釋号坡,但是沒有代碼分析宽堆,所以就自己上網尋找資料畜隶,一定要把這個Bean生命周期弄明白籽慢!

網上大部分都是驗證的Bean 在面試問的生命周期箱亿,其實查閱JDK還有一個完整的Bean生命周期届惋,這同時也驗證了書是具有片面性的驾茴,最fresh 的資料還是查閱原始JDK!!!

一、Bean 的完整生命周期

在傳統(tǒng)的Java應用中氢卡,bean的生命周期很簡單锈至,使用Java關鍵字 new 進行Bean 的實例化,然后該Bean 就能夠使用了译秦。一旦bean不再被使用峡捡,則由Java自動進行垃圾回收。

相比之下筑悴,Spring管理Bean的生命周期就復雜多了们拙,正確理解Bean 的生命周期非常重要阁吝,因為Spring對Bean的管理可擴展性非常強砚婆,下面展示了一個Bean的構造過程

Bean 的生命周期

如上圖所示,Bean 的生命周期還是比較復雜的突勇,下面來對上圖每一個步驟做文字描述:

Spring啟動装盯,查找并加載需要被Spring管理的bean,進行Bean的實例化

Bean實例化后對將Bean的引入和值注入到Bean的屬性中

如果Bean實現(xiàn)了BeanNameAware接口的話甲馋,Spring將Bean的Id傳遞給setBeanName()方法

如果Bean實現(xiàn)了BeanFactoryAware接口的話埂奈,Spring將調用setBeanFactory()方法,將BeanFactory容器實例傳入

如果Bean實現(xiàn)了ApplicationContextAware接口的話定躏,Spring將調用Bean的setApplicationContext()方法账磺,將bean所在應用上下文引用傳入進來。

如果Bean實現(xiàn)了BeanPostProcessor接口痊远,Spring就將調用他們的postProcessBeforeInitialization()方法垮抗。

如果Bean 實現(xiàn)了InitializingBean接口,Spring將調用他們的afterPropertiesSet()方法碧聪。類似的冒版,如果bean使用init-method聲明了初始化方法,該方法也會被調用

如果Bean 實現(xiàn)了BeanPostProcessor接口矾削,Spring就將調用他們的postProcessAfterInitialization()方法壤玫。

此時豁护,Bean已經準備就緒哼凯,可以被應用程序使用了。他們將一直駐留在應用上下文中楚里,直到應用上下文被銷毀断部。

如果bean實現(xiàn)了DisposableBean接口,Spring將調用它的destory()接口方法班缎,同樣蝴光,如果bean使用了destory-method 聲明銷毀方法她渴,該方法也會被調用。

上面是Spring 中Bean的核心接口和生命周期蔑祟,面試回答上述過程已經足夠了趁耗。但是翻閱JavaDoc文檔發(fā)現(xiàn)除了以上接口外,還有另外的初始化過程涉及的接口:摘自org.springframework.beans.factory.BeanFactory疆虚, 全部相關接口如下苛败,上述已有的就不用著重標注,把額外的相關接口著重標注下

Bean 完整的生命周期

文字解釋如下:

————————————初始化————————————

BeanNameAware.setBeanName() 在創(chuàng)建此bean的bean工廠中設置bean的名稱径簿,在普通屬性設置之后調用罢屈,在InitializinngBean.afterPropertiesSet()方法之前調用

BeanClassLoaderAware.setBeanClassLoader(): 在普通屬性設置之后,InitializingBean.afterPropertiesSet()之前調用

BeanFactoryAware.setBeanFactory() : 回調提供了自己的bean實例工廠篇亭,在普通屬性設置之后缠捌,在InitializingBean.afterPropertiesSet()或者自定義初始化方法之前調用

EnvironmentAware.setEnvironment(): 設置environment在組件使用時調用

EmbeddedValueResolverAware.setEmbeddedValueResolver(): 設置StringValueResolver 用來解決嵌入式的值域問題

ResourceLoaderAware.setResourceLoader(): 在普通bean對象之后調用,在afterPropertiesSet 或者自定義的init-method 之前調用译蒂,在 ApplicationContextAware 之前調用曼月。

ApplicationEventPublisherAware.setApplicationEventPublisher(): 在普通bean屬性之后調用,在初始化調用afterPropertiesSet 或者自定義初始化方法之前調用蹂随。在 ApplicationContextAware 之前調用十嘿。

MessageSourceAware.setMessageSource(): 在普通bean屬性之后調用,在初始化調用afterPropertiesSet 或者自定義初始化方法之前調用岳锁,在 ApplicationContextAware 之前調用绩衷。

ApplicationContextAware.setApplicationContext(): 在普通Bean對象生成之后調用,在InitializingBean.afterPropertiesSet之前調用或者用戶自定義初始化方法之前激率。在ResourceLoaderAware.setResourceLoader咳燕,ApplicationEventPublisherAware.setApplicationEventPublisher,MessageSourceAware之后調用乒躺。

ServletContextAware.setServletContext(): 運行時設置ServletContext招盲,在普通bean初始化后調用,在InitializingBean.afterPropertiesSet之前調用嘉冒,在 ApplicationContextAware 之后調用注:是在WebApplicationContext 運行時

BeanPostProcessor.postProcessBeforeInitialization() : 將此BeanPostProcessor 應用于給定的新bean實例 在任何bean初始化回調方法(像是InitializingBean.afterPropertiesSet或者自定義的初始化方法)之前調用曹货。這個bean將要準備填充屬性的值。返回的bean示例可能被普通對象包裝讳推,默認實現(xiàn)返回是一個bean顶籽。

BeanPostProcessor.postProcessAfterInitialization() : 將此BeanPostProcessor 應用于給定的新bean實例 在任何bean初始化回調方法(像是InitializingBean.afterPropertiesSet或者自定義的初始化方法)之后調用。這個bean將要準備填充屬性的值银觅。返回的bean示例可能被普通對象包裝

InitializingBean.afterPropertiesSet(): 被BeanFactory在設置所有bean屬性之后調用(并且滿足BeanFactory 和 ApplicationContextAware)礼饱。

————————————銷毀————————————

在BeanFactory 關閉的時候,Bean的生命周期會調用如下方法:

DestructionAwareBeanPostProcessor.postProcessBeforeDestruction(): 在銷毀之前將此BeanPostProcessor 應用于給定的bean實例。能夠調用自定義回調镊绪,像是DisposableBean 的銷毀和自定義銷毀方法匀伏,這個回調僅僅適用于工廠中的單例bean(包括內部bean)

實現(xiàn)了自定義的destory()方法

二、Bean 的生命周期驗證

為了驗證Bean生命周期的過程蝴韭,有兩種形式:一種是為面試而準備的够颠,一種是為了解全過程而準備的,下面來看代碼:

Book.class

public class Book implements BeanNameAware,BeanFactoryAware,

? ? ? ? ApplicationContextAware,InitializingBean,DisposableBean {

? ? private String bookName;

? ? public Book(){

? ? ? ? System.out.println("Book Initializing ");

? ? }

? ? public void setBeanFactory(BeanFactory beanFactory) throws BeansException {

? ? ? ? System.out.println("Book.setBeanFactory invoke");

? ? }

? ? public void setBeanName(String name) {

? ? ? ? System.out.println("Book.setBeanName invoke");

? ? }

? ? public void destroy() throws Exception {

? ? ? ? System.out.println("Book.destory invoke");

? ? }

? ? public void afterPropertiesSet() throws Exception {

? ? ? ? System.out.println("Book.afterPropertiesSet invoke");

? ? }

? ? public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {

? ? ? ? System.out.println("Book.setApplicationContext invoke");

? ? }

? ? public String getBookName() {

? ? ? ? return bookName;

? ? }

? ? public void setBookName(String bookName) {

? ? ? ? this.bookName = bookName;

? ? ? ? System.out.println("setBookName: Book name has set.");

? ? }

? ? public void myPostConstruct(){

? ? ? ? System.out.println("Book.myPostConstruct invoke");

? ? }

? ? // 自定義初始化方法

? ? @PostConstruct

? ? public void springPostConstruct(){

? ? ? ? System.out.println("@PostConstruct");

? ? }

? ? public void myPreDestory(){

? ? ? ? System.out.println("Book.myPreDestory invoke");

? ? ? ? System.out.println("---------------destroy-----------------");

? ? }

? ? // 自定義銷毀方法

? ? @PreDestroy

? ? public void springPreDestory(){

? ? ? ? System.out.println("@PreDestory");

? ? }

? ? @Override

? ? protected void finalize() throws Throwable {

? ? ? ? System.out.println("------inside finalize-----");

? ? }

}


自定義實現(xiàn)BeanPostProcessor 的MyBeanPostProcessor:

public class MyBeanPostProcessor implements BeanPostProcessor {

? ? // 容器加載的時候會加載一些其他的bean榄鉴,會調用初始化前和初始化后方法

? ? // 這次只關注book(bean)的生命周期

? ? public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {

? ? ? ? if(bean instanceof Book){

? ? ? ? ? ? System.out.println("MyBeanPostProcessor.postProcessBeforeInitialization");

? ? ? ? }

? ? ? ? return bean;

? ? }

? ? public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {

? ? ? ? if(bean instanceof Book){

? ? ? ? ? ? System.out.println("MyBeanPostProcessor.postProcessAfterInitialization");

? ? ? ? }

? ? ? ? return bean;

? ? }

}


在resources 目錄下新建Bean-Lifecycle.xml

<beans xmlns="http://www.springframework.org/schema/beans"

? ? ? xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

? ? ? xmlns:context="http://www.springframework.org/schema/context"

? ? ? xsi:schemaLocation="http://www.springframework.org/schema/beans

http://www.springframework.org/schema/beans/spring-beans-2.5.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">

? ? ? <!-- 掃描bean -->

? ? <context:component-scan base-package="com.bean.lifecycle"/>

? ? <!-- 實現(xiàn)了用戶自定義初始化和銷毀方法 -->

? ? <bean id="book" class="com.bean.lifecycle.Book" init-method="myPostConstruct" destroy-method="myPreDestory">

? ? ? ? ? <!-- 注入bean 屬性名稱 -->

? ? ? ? <property name="bookName" value="thingking in java" />

? ? </bean>

? ? ? <!--引入自定義的BeanPostProcessor-->

? ? <bean class="com.bean.lifecycle.MyBeanPostProcessor"/>

</beans>


做一個啟動類的測試摧找,新建SpringBeanLifecycleApplication

public class SpringBeanLifecycleApplication {

? ? public static void main(String[] args) throws InterruptedException {

? ? ? ? // 為面試而準備的Bean生命周期加載過程

? ? ? ? ApplicationContext context = new ClassPathXmlApplicationContext("Bean-Lifecycle.xml");

? ? ? ? Book book = (Book)context.getBean("book");

? ? ? ? System.out.println("Book name = " + book.getBookName());

? ? ? ? ((ClassPathXmlApplicationContext) context).destroy();

? ? }

}


啟動測試,輸出結果如下:

Book Initializing

setBookName: Book name has set.

Book.setBeanName invoke

Book.setBeanFactory invoke

Book.setApplicationContext invoke

MyBeanPostProcessor.postProcessBeforeInitialization

@PostConstruct

Book.afterPropertiesSet invoke

Book.myPostConstruct invoke

MyBeanPostProcessor.postProcessAfterInitialization

Book name = thingking in java

@PreDestory

Book.destory invoke

Book.myPreDestory invoke

---------------destroy-----------------


為了驗證Bean完整的生命周期牢硅,需要新建一個SubBookClass 繼承Book類

public class SubBookClass extends Book implements BeanClassLoaderAware,

? ? ? ? EnvironmentAware,EmbeddedValueResolverAware,ResourceLoaderAware,

? ? ? ? ApplicationEventPublisherAware,MessageSourceAware{

? ? private String bookSystem;

? ? public String getBookSystem() {

? ? ? ? return bookSystem;

? ? }

? ? public void setBookSystem(String bookSystem) {

? ? ? ? System.out.println("設置BookSystem 的屬性值");

? ? ? ? this.bookSystem = bookSystem;

? ? }

? ? public void setBeanClassLoader(ClassLoader classLoader) {

? ? ? ? System.out.println("SubBookClass.setBeanClassLoader() 方法被調用了");

? ? }

? ? public void setApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher) {

? ? ? ? System.out.println("SubBookClass.setApplicationEventPublisher() 方法被調用了");

? ? }

? ? public void setEmbeddedValueResolver(StringValueResolver resolver) {

? ? ? ? System.out.println("SubBookClass.setEmbeddedValueResolver() 方法被調用了");

? ? }

? ? public void setEnvironment(Environment environment) {

? ? ? ? System.out.println("SubBookClass.setEnvironment() 方法被調用了");

? ? }

? ? public void setMessageSource(MessageSource messageSource) {

? ? ? ? System.out.println("SubBookClass.setMessageSource() 方法被調用了");

? ? }

? ? public void setResourceLoader(ResourceLoader resourceLoader) {

? ? ? ? System.out.println("SubBookClass.setResourceLoader() 方法被調用了");

? ? }

}


上述SubBookClass類與Book是互補關系蹬耘。

新建一個SubBean-Lifecycle.xml,注入SubBookClass

<beans xmlns="http://www.springframework.org/schema/beans"

? ? ? xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

? ? ? xmlns:context="http://www.springframework.org/schema/context"

? ? ? xsi:schemaLocation="http://www.springframework.org/schema/beans

http://www.springframework.org/schema/beans/spring-beans-2.5.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">

? ? <bean id="bookClass" class="com.bean.lifecycle.SubBookClass" init-method="myPostConstruct" destroy-method="myPreDestory">

? ? ? ? <property name="bookSystem" value="Java System" />

? ? </bean>

? ? <bean class="com.bean.lifecycle.MyBeanPostProcessor"/>

</beans>


完整的SpringBeanLifecycleApplication 如下:

public class SpringBeanLifecycleApplication {

? ? public static void main(String[] args) throws InterruptedException {

? ? ? ? // 為面試而準備的Bean生命周期加載過程

? ? ? ? ApplicationContext context = new ClassPathXmlApplicationContext("Bean-Lifecycle.xml");

? ? ? ? Book book = (Book)context.getBean("book");

? ? ? ? System.out.println("Book name = " + book.getBookName());

? ? ? ? ((ClassPathXmlApplicationContext) context).destroy();

? ? ? ? // 完整的加載過程减余,當然了解的越多越好

? ? ? ? ApplicationContext applicationContext = new ClassPathXmlApplicationContext("SubBean-Lifecycle.xml");

? ? ? ? SubBookClass subBookClass = (SubBookClass) applicationContext.getBean("bookClass");

? ? ? ? System.out.println("BookSystemName = " + subBookClass.getBookSystem());

? ? ? ? ((ClassPathXmlApplicationContext) applicationContext).registerShutdownHook();

? ? }

}


輸出完整的結果:

Book Initializing

setBookName: Book name has set.

Book.setBeanName invoke

Book.setBeanFactory invoke

Book.setApplicationContext invoke

MyBeanPostProcessor.postProcessBeforeInitialization

@PostConstruct

Book.afterPropertiesSet invoke

Book.myPostConstruct invoke

MyBeanPostProcessor.postProcessAfterInitialization

Book name = thingking in java

@PreDestory

Book.destory invoke

Book.myPreDestory invoke

---------------destroy-----------------

Book Initializing

設置BookSystem 的屬性值

Book.setBeanName invoke

SubBookClass.setBeanClassLoader() 方法被調用了

Book.setBeanFactory invoke

SubBookClass.setEnvironment() 方法被調用了

SubBookClass.setEmbeddedValueResolver() 方法被調用了

SubBookClass.setResourceLoader() 方法被調用了

SubBookClass.setApplicationEventPublisher() 方法被調用了

SubBookClass.setMessageSource() 方法被調用了

Book.setApplicationContext invoke

MyBeanPostProcessor.postProcessBeforeInitialization

Book.afterPropertiesSet invoke

Book.myPostConstruct invoke

MyBeanPostProcessor.postProcessAfterInitialization

BookSystemName = Java System

Book.destory invoke

Book.myPreDestory invoke

---------------destroy-----------------


后記:這篇文章是我翻閱各種書籍和從網上查找資料综苔,包括國外一些網站從而得到的結論,記錄下來位岔,但是我沒有發(fā)現(xiàn)Spring Bean的生命周期(非常詳細)這篇文章中InstantiationAwareBeanPostProcessorAdapter 這個類和工廠后置處理器接口方法如筛,知道的朋友歡迎指教,感謝抒抬。

歡迎工作一到五年的Java工程師朋友們加入Java程序員開發(fā): 721575865

群內提供免費的Java架構學習資料(里面有高可用杨刨、高并發(fā)、高性能及分布式擦剑、Jvm性能調優(yōu)妖胀、Spring源碼斩松,MyBatis皆尔,Netty,Redis,Kafka,Mysql,Zookeeper,Tomcat,Docker,Dubbo,Nginx等多個知識點的架構資料)合理利用自己每一分每一秒的時間來學習提升自己,不要再用"沒有時間“來掩飾自己思想上的懶惰呻澜!趁年輕纠屋,使勁拼涂臣,給未來的自己一個交代!

?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末售担,一起剝皮案震驚了整個濱河市赁遗,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌族铆,老刑警劉巖岩四,帶你破解...
    沈念sama閱讀 207,248評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異骑素,居然都是意外死亡炫乓,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,681評論 2 381
  • 文/潘曉璐 我一進店門献丑,熙熙樓的掌柜王于貴愁眉苦臉地迎上來末捣,“玉大人,你說我怎么就攤上這事创橄÷嶙觯” “怎么了?”我有些...
    開封第一講書人閱讀 153,443評論 0 344
  • 文/不壞的土叔 我叫張陵妥畏,是天一觀的道長邦邦。 經常有香客問我,道長醉蚁,這世上最難降的妖魔是什么燃辖? 我笑而不...
    開封第一講書人閱讀 55,475評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮网棍,結果婚禮上黔龟,老公的妹妹穿的比我還像新娘。我一直安慰自己滥玷,他們只是感情好氏身,可當我...
    茶點故事閱讀 64,458評論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著惑畴,像睡著了一般蛋欣。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上如贷,一...
    開封第一講書人閱讀 49,185評論 1 284
  • 那天陷虎,我揣著相機與錄音,去河邊找鬼杠袱。 笑死泻红,一個胖子當著我的面吹牛,可吹牛的內容都是我干的霞掺。 我是一名探鬼主播谊路,決...
    沈念sama閱讀 38,451評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼菩彬!你這毒婦竟也來了缠劝?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 37,112評論 0 261
  • 序言:老撾萬榮一對情侶失蹤骗灶,失蹤者是張志新(化名)和其女友劉穎惨恭,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體耙旦,經...
    沈念sama閱讀 43,609評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡脱羡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,083評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片锉罐。...
    茶點故事閱讀 38,163評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡帆竹,死狀恐怖,靈堂內的尸體忽然破棺而出脓规,到底是詐尸還是另有隱情栽连,我是刑警寧澤,帶...
    沈念sama閱讀 33,803評論 4 323
  • 正文 年R本政府宣布侨舆,位于F島的核電站秒紧,受9級特大地震影響,放射性物質發(fā)生泄漏挨下。R本人自食惡果不足惜熔恢,卻給世界環(huán)境...
    茶點故事閱讀 39,357評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望臭笆。 院中可真熱鬧绩聘,春花似錦、人聲如沸耗啦。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,357評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽帜讲。三九已至衅谷,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間似将,已是汗流浹背获黔。 一陣腳步聲響...
    開封第一講書人閱讀 31,590評論 1 261
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留在验,地道東北人玷氏。 一個月前我還...
    沈念sama閱讀 45,636評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像腋舌,于是被迫代替她去往敵國和親盏触。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 42,925評論 2 344

推薦閱讀更多精彩內容