Spring 中bean 的生命週期短暫嗎?面試
在spring中,從BeanFactory或ApplicationContext取得的實例爲Singleton,也就是預設爲每個Bean的別名只能維持一個實例,而不是每次都產生一個新的對象使用Singleton模式產生單一實例,對單線程的程序說並不會有什麼問題,但對於多線程的程序,就必須注意安全(Thread-safe)的議題,防止多個線程同時存取共享資源所引起的數據不一樣步問題。spring
然而在spring中 能夠設定每次從BeanFactory或ApplicationContext指定別名並取得Bean時都產生一個新的實例:例如:緩存
在spring中,singleton屬性默認是true,只有設定爲false,則每次指定別名取得的Bean時都會產生一個新的實例安全
一個Bean從建立到銷燬,若是是用BeanFactory來生成,管理Bean的話,會經歷幾個執行階段(如圖1.1):多線程
1:Bean的創建:框架
容器尋找Bean的定義信息並將其實例化。post
2:屬性注入:spa
使用依賴注入,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也相似,以下
一、實例化一個Bean--也就是咱們常說的new;
二、按照Spring上下文對實例化的Bean進行配置--也就是IOC注入;
三、若是這個Bean已經實現了BeanNameAware接口,會調用它實現的setBeanName(String)方法,此處傳遞的就是Spring配置文件中Bean的id值
四、若是這個Bean已經實現了BeanFactoryAware接口,會調用它實現的setBeanFactory(setBeanFactory(BeanFactory)傳遞的是Spring工廠自身(能夠用這個方式來獲取其它Bean,只需在Spring配置文件中配置一個普通的Bean就能夠);
五、若是這個Bean已經實現了ApplicationContextAware接口,會調用setApplicationContext(ApplicationContext)方法,傳入Spring上下文(一樣這個方式也能夠實現步驟4的內容,但比4更好,由於ApplicationContext是BeanFactory的子接口,有更多的實現方法);
六、若是這個Bean關聯了BeanPostProcessor接口,將會調用postProcessBeforeInitialization(Object obj, String s)方法,BeanPostProcessor常常被用做是Bean內容的更改,而且因爲這個是在Bean初始化結束時調用那個的方法,也能夠被應用於內存或緩存技術;
七、若是Bean在Spring配置文件中配置了init-method屬性會自動調用其配置的初始化方法。
八、若是這個Bean關聯了BeanPostProcessor接口,將會調用postProcessAfterInitialization(Object obj, String s)方法、;
注:以上工做完成之後就能夠應用這個Bean了,那這個Bean是一個Singleton的,因此通常狀況下咱們調用同一個id的Bean會是在內容地址相同的實例,固然在Spring配置文件中也能夠配置非Singleton,這裏咱們不作贅述。
九、當Bean再也不須要時,會通過清理階段,若是Bean實現了DisposableBean這個接口,會調用那個其實現的destroy()方法;
十、最後,若是這個Bean的Spring配置中配置了destroy-method屬性,會自動調用其配置的銷燬方法。
以上10步驟能夠做爲面試或者筆試的模板,另外咱們這裏描述的是應用Spring上下文Bean的生命週期,若是應用Spring的工廠也就是BeanFactory的話去掉第5步就Ok了。
由BeanFactory讀取Bean定義文件,並生成各個實例。
執行Bean的屬性依賴注入。
若是Bean類實現了org.springframework.beans.factory.BeanNameAware接口,則執行其setBeanName()方法。
若是Bean類實現了org.springframework.beans.factory.BeanFactoryAware接口,則執行其setBeanFactory()方法。
容器中若是有實現org.springframework.beans.factory.BeanPostProcessors接口的實例,則任何Bean在初始化以前都會執行這個實例的processBeforeInitialization()方法。
若是Bean類實現了org.springframework.beans.factory.InitializingBean接口,則執行其afterPropertiesSet()方法。
在Bean定義文件中使用「init-method」屬性設定方法名稱,以下:
<bean id="demoBean" class="com.yangsq.bean.DemoBean" init-method="initMethod"> |
這時會執行initMethod()方法,注意,這個方法是不帶參數的。
容器中若是有實現org.springframework.beans.factory.BeanPostProcessors接口的實例,則任何Bean在初始化以前都會執行這個實例的processAfterInitialization()方法。
在容器關閉時,若是Bean類實現了org.springframework.beans.factory.DisposableBean接口,則執行它的destroy()方法。
在容器關閉時,能夠在Bean定義文件中使用「destory-method」定義的方法
<bean id="demoBean" class="com.yangsq.bean.DemoBean" destory-method="destroyMethod"> |
這時會執行destroyMethod()方法,注意,這個方法是不帶參數的。
以上就是BeanFactory維護的一個Bean的生命週期。下面這個圖可能更直觀一些:
若是使用ApplicationContext來維護一個Bean的生命週期,則基本上與上邊的流程相同,只不過在執行BeanNameAware的setBeanName()後,如有Bean類實現了org.springframework.context.ApplicationContextAware接口,則執行其setApplicationContext()方法,而後再進行BeanPostProcessors的processBeforeInitialization()