Spring 中bean 的生命周期短暫嗎?
在spring中,從BeanFactory或ApplicationContext取得的執行個體為Singleton,也就是預設為每一個Bean的别名隻能維持一個執行個體,而不是每次都産生一個新的對象使用Singleton模式産生單一執行個體,對單線程的程式說并不會有什麼問題,但對于多線程的程式,就必須注意安全(Thread-safe)的議題,防止多個線程同時存取共享資源所引發的資料不同步問題。
然而在spring中 可以設定每次從BeanFactory或ApplicationContext指定别名并取得Bean時都産生一個新的執行個體:例如:
在spring中,singleton屬性預設是true,隻有設定為false,則每次指定别名取得的Bean時都會産生一個新的執行個體
一個Bean從建立到銷毀,如果是用BeanFactory來生成,管理Bean的話,會經曆幾個執行階段(如圖1.1):

1:Bean的建立:
容器尋找Bean的定義資訊并将其執行個體化。
2:屬性注入:
使用依賴注入,Spring按照Bean定義資訊配置Bean所有屬性
3:BeanNameAware的setBeanName():
如果Bean類有實作org.springframework.beans.BeanNameAware接口,工廠調用Bean的setBeanName()方法傳遞Bean的ID。
4:BeanFactoryAware的setBeanFactory():
如果Bean類有實作org.springframework.beans.factory.BeanFactoryAware接口,工廠調用setBeanFactory()方法傳入工廠自身。
5:BeanPostProcessors的ProcessBeforeInitialization()
如果有org.springframework.beans.factory.config.BeanPostProcessors和Bean關聯,那麼其postProcessBeforeInitialization()方法将被将被調用。
6:initializingBean的afterPropertiesSet():
如果Bean類已實作org.springframework.beans.factory.InitializingBean接口,則執行他的afterProPertiesSet()方法
7:Bean定義檔案中定義init-method:
可以在Bean定義檔案中使用"init-method"屬性設定方法名稱例如:
如果有以上設定的話,則執行到這個階段,就會執行initBean()方法
8:BeanPostProcessors的ProcessaAfterInitialization()
如果有任何的BeanPostProcessors執行個體與Bean執行個體關聯,則執行BeanPostProcessors執行個體的ProcessaAfterInitialization()方法
此時,Bean已經可以被應用系統使用,并且将保留在BeanFactory中知道它不在被使用。有兩種方法可以将其從BeanFactory中删除掉(如圖1.2):
1:DisposableBean的destroy()
在容器關閉時,如果Bean類有實作org.springframework.beans.factory.DisposableBean接口,則執行他的destroy()方法
2:Bean定義檔案中定義destroy-method
在容器關閉時,可以在Bean定義檔案中使用"destroy-method"屬性設定方法名稱,例如:
如果有以上設定的話,則進行至這個階段時,就會執行destroy()方法,如果是使用ApplicationContext來生成并管理Bean的話則稍有不同,使用ApplicationContext來生成及管理Bean執行個體的話,在執行BeanFactoryAware的setBeanFactory()階段後,若Bean類上有實作org.springframework.context.ApplicationContextAware接口,則執行其setApplicationContext()方法,接着才執行BeanPostProcessors的ProcessBeforeInitialization()及之後的流程。
找工作的時候有些人會被問道Spring中Bean的生命周期,其實也就是考察一下對Spring是否熟悉,工作中很少用到其中的内容,那我們簡單看一下。
在說明前可以思考一下Servlet的生命周期:執行個體化,初始init,接收請求service,銷毀destroy;
Spring上下文中的Bean也類似,如下
1、執行個體化一個Bean--也就是我們常說的new;
2、按照Spring上下文對執行個體化的Bean進行配置--也就是IOC注入;
3、如果這個Bean已經實作了BeanNameAware接口,會調用它實作的setBeanName(String)方法,此處傳遞的就是Spring配置檔案中Bean的id值
4、如果這個Bean已經實作了BeanFactoryAware接口,會調用它實作的setBeanFactory(setBeanFactory(BeanFactory)傳遞的是Spring工廠自身(可以用這個方式來擷取其它Bean,隻需在Spring配置檔案中配置一個普通的Bean就可以);
5、如果這個Bean已經實作了ApplicationContextAware接口,會調用setApplicationContext(ApplicationContext)方法,傳入Spring上下文(同樣這個方式也可以實作步驟4的内容,但比4更好,因為ApplicationContext是BeanFactory的子接口,有更多的實作方法);
6、如果這個Bean關聯了BeanPostProcessor接口,将會調用postProcessBeforeInitialization(Object obj, String s)方法,BeanPostProcessor經常被用作是Bean内容的更改,并且由于這個是在Bean初始化結束時調用那個的方法,也可以被應用于記憶體或緩存技術;
7、如果Bean在Spring配置檔案中配置了init-method屬性會自動調用其配置的初始化方法。
8、如果這個Bean關聯了BeanPostProcessor接口,将會調用postProcessAfterInitialization(Object obj, String s)方法、;
注:以上工作完成以後就可以應用這個Bean了,那這個Bean是一個Singleton的,是以一般情況下我們調用同一個id的Bean會是在内容位址相同的執行個體,當然在Spring配置檔案中也可以配置非Singleton,這裡我們不做贅述。
9、當Bean不再需要時,會經過清理階段,如果Bean實作了DisposableBean這個接口,會調用那個其實作的destroy()方法;
10、最後,如果這個Bean的Spring配置中配置了destroy-method屬性,會自動調用其配置的銷毀方法。
以上10步驟可以作為面試或者筆試的模闆,另外我們這裡描述的是應用Spring上下文Bean的生命周期,如果應用Spring的工廠也就是BeanFactory的話去掉第5步就Ok了。
這Spring架構中,一旦把一個bean納入到Spring IoC容器之中,這個bean的生命周期就會交由容器進行管理,一般擔當管理者角色的是BeanFactory或ApplicationContext。認識一下Bean的生命周期活動,對更好的利用它有很大的幫助。
下面以BeanFactory為例,說明一個Bean的生命周期活動:
- Bean的建立
由BeanFactory讀取Bean定義檔案,并生成各個執行個體。
- Setter注入
執行Bean的屬性依賴注入。
- BeanNameAware的setBeanName()
如果Bean類實作了org.springframework.beans.factory.BeanNameAware接口,則執行其setBeanName()方法。
- BeanFactoryAware的setBeanFactory()
如果Bean類實作了org.springframework.beans.factory.BeanFactoryAware接口,則執行其setBeanFactory()方法。
- BeanPostProcessors的processBeforeInitialization()
容器中如果有實作org.springframework.beans.factory.BeanPostProcessors接口的執行個體,則任何Bean在初始化之前都會執行這個執行個體的processBeforeInitialization()方法。
- InitializingBean的afterPropertiesSet()
如果Bean類實作了org.springframework.beans.factory.InitializingBean接口,則執行其afterPropertiesSet()方法。
- Bean定義檔案中定義init-method
在Bean定義檔案中使用“init-method”屬性設定方法名稱,如下:
<bean id="demoBean" class="com.yangsq.bean.DemoBean" init-method="initMethod"> ....... </bean> |
這時會執行initMethod()方法,注意,這個方法是不帶參數的。
- BeanPostProcessors的processAfterInitialization()
容器中如果有實作org.springframework.beans.factory.BeanPostProcessors接口的執行個體,則任何Bean在初始化之前都會執行這個執行個體的processAfterInitialization()方法。
- DisposableBean的destroy()
在容器關閉時,如果Bean類實作了org.springframework.beans.factory.DisposableBean接口,則執行它的destroy()方法。
- Bean定義檔案中定義destroy-method
在容器關閉時,可以在Bean定義檔案中使用“destory-method”定義的方法
<bean id="demoBean" class="com.yangsq.bean.DemoBean" destory-method="destroyMethod"> </bean> |
這時會執行destroyMethod()方法,注意,這個方法是不帶參數的。
以上就是BeanFactory維護的一個Bean的生命周期。下面這個圖可能更直覺一些:
如果使用ApplicationContext來維護一個Bean的生命周期,則基本上與上邊的流程相同,隻不過在執行BeanNameAware的setBeanName()後,若有Bean類實作了org.springframework.context.ApplicationContextAware接口,則執行其setApplicationContext()方法,然後再進行BeanPostProcessors的processBeforeInitialization()
實際上,ApplicationContext除了向BeanFactory那樣維護容器外,還提供了更加豐富的架構功能,如Bean的消息,事件處理機制等。