JAVA面試題:Spring中bean的生命週期

a1322674015發表於2019-11-26

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的訊息,事件處理機制等。


來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/69946034/viewspace-2665799/,如需轉載,請註明出處,否則將追究法律責任。

相關文章