IoC基礎篇(一)--- Spring容器中Bean的生命周期


日出日落,春去秋來,花隨流水,北雁南飛,世間萬物皆有生死輪回。從調用XML中的Bean配置信息,到應用到具體實例中,再到銷毀,Bean也有屬於它的生命周期。

 

人類大腦對圖像的認知能力永遠高於文字,因此,閑言少敘,書歸正傳,上圖先:

步驟很多,切莫驚慌,我們可以把上面的步驟歸納如下:

1-2:創建實例;

  現在假設spring就是個容器,而配置文件中配置的bean屬性才是我們真正需要的東西。創建實例就是說,我把配置文件中的bean信息取出來化作一個真正的bean並放到容器中。

3-4:注入依賴關系;

  第3步是創建實例之后對實例作了一些處理,第4步是把xml中配置的bean屬性值賦予給容器中的實例化之后的bean。

5:bean初始化之前的處理;

  應用開發者需要把容器中實例化的bean拿出來用,這個拿出來的過程就是初始化(注意實例化與初始化的區別,instantiation 和initialization,分得清嗎?英語沒學好怪我咯?),第五步就是在初始化之前,對已經實例化的bean再作一定的處理。

6,7:初始化。

  如果bean實現了InitializingBean,那么將調用InitializingBean的afterPropertiesSet()方法做一些初始化處理。如果沒有實現InitializingBean,而是在配置文件中定義了init-method屬性值,那么系統會找到init-method對應的方法並執行之,程序猿哥哥一般在這個方法里寫一些初始化操作;

8:bean初始化之后的處理。

  初始化之后在這個方法中再對bean進行修飾裝點。

9,10:交給應用開發人員處理;

  如果在<bean>中指定Bean的作用范圍是scopt="prototype",那么系統將bean返回給調用者,spring就不管了(如果兩個實例調用的話,每一次調用都要重新初始化,一個實例的修改不會影響另一個實例的值。如果指定Bean的作用范圍是scope="singleton",則把bean放到緩沖池中,並將bean的引用返回給調用者。這個時候,如果兩個實例調用的話,因為它們用的是同一個引用,任何一方的修改都會影響到另一方。)

11.bean用完之后;

  對於scope="singleton"的bean,使用完之后spring容器會做一些處理,比如編寫釋放資源、記錄日志等操作。

12.銷毀;

  調用配置文件中的銷毀方法銷毀實例。

 

光說不練假把式。來看實例:

<?xml version="1.0" encoding="UTF-8" ?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
    xsi:schemaLocation="http://www.springframework.org/schema/beans 
       http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
    <bean id="jay" class="com.mesopotamia.bean_life_cycle.JayChou" 
         init-method="myInit"
         destroy-method="myDestory"
         p:sex="男"
         p:girlFriend="蔡依林" 
         p:profession="歌手,演員,導演,主持"
         scope="singleton"
         /> 
</beans>

上面是配置文件,取名為jayConfig.xml,注意bean中定義了init-method、destroy-method屬性,翻閱上文查看這兩個屬性的作用。該bean還定義了scope屬性,對應第9步。

在BeanFactory的bean生命周期中,第1、3、5、8步是由程序員自己擴展的。自己擴展類並在代碼中注冊使用。

下面看一個例子:

package com.mesopotamia.bean_life_cycle;

import java.beans.PropertyDescriptor;

import org.springframework.beans.BeansException;
import org.springframework.beans.PropertyValues;
import org.springframework.beans.factory.config.InstantiationAwareBeanPostProcessorAdapter;

public class MyInstantiationAwareBeanPostProcessor extends InstantiationAwareBeanPostProcessorAdapter{

    public Object postProcessBeforeInstantiation(Class beanClass, String beanName) throws BeansException {
        if("jay".equals(beanName)){
            System.out.println("MyInstantiationAwareBeanPostProcessor.postProcessBeforeInstantiation");            
        }        
        return null;
    }
    /**實例化Bean后,對Bean進行梳妝打扮*/
    public boolean postProcessAfterInstantiation(Object bean, String beanName) throws BeansException {
        if("jay".equals(beanName)){
        System.out.println("InstantiationAwareBeanPostProcessor.postProcessAfterInstantiation");
        }
        return true;
    }

    //把Bean的配置值賦值給Bean實例的屬性。
    public PropertyValues postProcessPropertyValues(
            PropertyValues pvs, PropertyDescriptor[] pds, Object bean, String beanName)
            throws BeansException {
        if("jay".equals(beanName)){
           System.out.println("InstantiationAwareBeanPostProcessor.postProcessPropertyValues");
        }
        return pvs;
    }

    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        return bean;
    }

    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        return bean;
    }
}

上面的類是繼承了InstantiationAwareBeanPostProcessorAdopter的,而InstantiationAwareBeanPostProcessorAdopter這個適配器與是InstantiationAwareBeanPostProcessor的擴展類。

package com.mesopotamia.bean_life_cycle;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;


public class MyBeanPostProcessor implements BeanPostProcessor{
    
    private Log log=LogFactory.getLog(MyBeanPostProcessor.class);

    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {        
        if(beanName.equals("jay")){
            JayChou jay = (JayChou)bean;
            log.info("JayChou當前的女朋友是:"+jay.getGirlFriend());
            jay.setGirlFriend("昆凌");
            log.info("調用BeanPostProcessor的postProcessAfterInitialization處理后," +
                    "JayChou的女朋友變成:"+jay.getGirlFriend());
        }
        return bean;

    }

    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {        
        if(beanName.equals("jay")){
            JayChou jay = (JayChou)bean;
            log.info("配置文件中Jay的女朋友是:"+jay.getGirlFriend());
            jay.setGirlFriend("侯佩岑");
            log.info("調用BeanPostProcessor的postProcessBeforeInitialization處理后," +
                    "Jay的女朋友變為:"+jay.getGirlFriend());
            
        }
        return bean;
    }
}

上面的代碼實現了BeanPostProcessor。InstantiationAwareBeanPostProcessor是實例化前后做的事情,BeanPostProcessor是初始化前后做的事情,它們之間應該存在着父子關系吧?當然是,從名字就能看出來前者是后者的擴展類,讀者可以自己下載spring源碼查看。這兩個類定義好了需要注冊才能使用(把它們注冊到spring容器中使用),稍后再講。這里先講bean類:

package com.mesopotamia.bean_life_cycle;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.beans.factory.DisposableBean;
import org.springframework.beans.factory.InitializingBean;

public class JayChou implements InitializingBean,DisposableBean {
    
    private static Log log=LogFactory.getLog(JayChou.class);
    private String sex;
    private String profession;
    private String girlFriend;
    
    

    public JayChou(){
        log.info("調用JayChou的無參構造函數。傑倫出道啦。");
    }

    public String getSex() {
        return sex;
    }

    public void setSex(String sex) {
        this.sex = sex;
    }
    

    public void setGirlFriend(String girlFriend) {
        this.girlFriend = girlFriend;
    }

    public String getGirlFriend() {
        return girlFriend;
    }

    public String getProfession() {
        return profession;
    }

    public void setProfession(String profession) {
        this.profession = profession;
    }


    public void afterPropertiesSet() throws Exception {
        this.profession="歌手";
        log.info("調用InitializingBean.afterPropertiesSet()," +
                "屬性配置完畢了再做些善后工作。");
        
    }

    public void destroy() throws Exception {
        log.info("調用DisposableBean.destory(),銷毀。。");
        
    }
    
    public void myInit() {        
        this.girlFriend = "徐若瑄";
        log.info("通過調用配置文件初始化女朋友為:"+this.girlFriend);
    }

    public void myDestory() {
        System.out.println("調用myDestroy()。");
    }
    
    public String toString(){
        return "JayChou簡介:" +
                "    性別:"+sex+
                "    職業:"+profession+
                "    女朋友:"+girlFriend;
    }


}

這個bean實現了InitializingBean,DisposableBean,那么第六步和第11步就可以用得到了。

下面看Main方法類:

package com.mesopotamia.bean_life_cycle;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.config.ConfigurableBeanFactory;
import org.springframework.beans.factory.xml.XmlBeanFactory;
import org.springframework.core.io.ClassPathResource;
import org.springframework.core.io.Resource;


public class BeanLifeCycleMain {
    
    private static Log log=LogFactory.getLog(BeanLifeCycleMain.class);
    
    private static void LifeCycleInBeanFactory(){
        Resource res = new ClassPathResource("com/mesopotamia/bean_life_cycle/jayConfig.xml");    
       BeanFactory bf = new XmlBeanFactory(res);
     
       /**BeanFactory是ConfigurableBeanFactory的父類,且該父類沒有這個添加方法,所以要轉換。*/
       ((ConfigurableBeanFactory)bf).addBeanPostProcessor(new MyInstantiationAwareBeanPostProcessor()); 
       //BeanPostProcessor
       ((ConfigurableBeanFactory)bf).addBeanPostProcessor(new MyBeanPostProcessor());             
       
       JayChou jay = (JayChou)bf.getBean("jay");
       log.info("jay:"+jay.toString());
      
      // jay.setGirlFriend("溫嵐");
       JayChou jay2 = bf.getBean("jay",JayChou.class);
       jay2.setGirlFriend("溫嵐");
       log.info("jay2:"+jay2.toString());
       log.info("jay:"+jay.toString());
       
       log.info(jay == jay2);
       ((XmlBeanFactory)bf).destroySingletons();
    }
    public static void main(String[] args) {
        LifeCycleInBeanFactory();
    }
}

第17、18行是裝載配置文件並啟動容器(相關知識請自己腦補)。第21行和23行是注冊事件,注冊后才能正常使用。先看一下執行結果:

1  2015-11-07 22:01:26,907  INFO [main] (XmlBeanDefinitionReader.java:315) - Loading XML bean definitions from class path resource [com/mesopotamia/bean_life_cycle/jayConfig.xml]
2  MyInstantiationAwareBeanPostProcessor.postProcessBeforeInstantiation
3  2015-11-07 22:01:27,068  INFO [main] (JayChou.java:18) - 調用JayChou的無參構造函數。傑倫出道啦。
4  InstantiationAwareBeanPostProcessor.postProcessAfterInstantiation
5  InstantiationAwareBeanPostProcessor.postProcessPropertyValues
6  2015-11-07 22:01:27,106  INFO [main] (MyBeanPostProcessor.java:28) - 配置文件中Jay的女朋友是:蔡依林
7  2015-11-07 22:01:27,106  INFO [main] (MyBeanPostProcessor.java:30) - 調用BeanPostProcessor的postProcessBeforeInitialization處理后,Jay的女朋友變為:侯佩岑
8  2015-11-07 22:01:27,106  INFO [main] (JayChou.java:49) - 調用InitializingBean.afterPropertiesSet(),屬性配置完畢了再做些善后工作。
9  2015-11-07 22:01:27,106  INFO [main] (JayChou.java:61) - 通過調用配置文件初始化女朋友為:徐若瑄
10  2015-11-07 22:01:27,106  INFO [main] (MyBeanPostProcessor.java:16) - JayChou當前的女朋友是:徐若瑄
11  2015-11-07 22:01:27,107  INFO [main] (MyBeanPostProcessor.java:18) - 調用BeanPostProcessor的postProcessAfterInitialization處理后,JayChou的女朋友變成:昆凌
12  2015-11-07 22:01:27,110  INFO [main] (BeanLifeCycleMain.java:26) - jay:JayChou簡介:   性別:男    職業:歌手   女朋友:昆凌
13  2015-11-07 22:01:27,110  INFO [main] (BeanLifeCycleMain.java:31) - jay2:JayChou簡介:  性別:男    職業:歌手   女朋友:溫嵐
14  2015-11-07 22:01:27,110  INFO [main] (BeanLifeCycleMain.java:32) - jay:JayChou簡介:   性別:男    職業:歌手   女朋友:溫嵐
15  2015-11-07 22:01:27,110  INFO [main] (BeanLifeCycleMain.java:34) - true
16  2015-11-07 22:01:27,110  INFO [main] (DefaultSingletonBeanRegistry.java:422) - Destroying singletons in org.springframework.beans.factory.xml.XmlBeanFactory@6e293a: defining beans [jay]; root of factory hierarchy
17  2015-11-07 22:01:27,111  INFO [main] (JayChou.java:55) - 調用DisposableBean.destory(),銷毀。。
18  調用myDestroy()。

行1:加載xml。

  行2:實例化之前處理了一些事情。

  行3:duang!開始實例化,實例化當然首先要執行構造函數(這是美好世界的窗口)。

  行4:實例化之后處理了一些事情。

  行5:實例化之后注入屬性值之前要調用這個函數。

  行6:注入xml的屬性值,可以看到是配置中的"蔡依林"。

  行7:注入屬性值后,隨即又把女朋友這個屬性的值改為了"侯佩岑"。

  行8:屬性配置完畢了做一些善后工作。在JayChou類的第48行可以看到,我把職業改為了"歌手",所以后面顯示的職業都是"歌手",而不是配置文件中的"歌手,演員,導演,主持"。

  行9:調用了配置文件中的init-method。通過<bean>的class屬性找到這個類,再找到屬性值對應的這個方法執行。行8和行9對應流程圖表中的第六步和第七步,在此可以看到先執行第6步,后執行第七步。然而,行9是在配置文件中實現的,這正符合spring容器的宗旨,而行8的實現必須implements InitializingBean,給代碼帶來復雜度和污染,因此行8(也就是圖表第六步)一般是不提倡的。

  行10和行11是初始化之后做的事情,首先顯示當前女朋友,在行9中已經改為徐若瑄。緊接着行11又把女朋友改為了昆凌。

  行12-行15:看對應的代碼就會發現,兩個對象其實引用的是同一個地址,jay2修改了屬性之后jay1也會跟着作改變,這就是singleton配置方式的作用。

  行16是關閉容器。

  行17和行18,一個是調用DisposableBean的銷毀,一個是調用配置文件的銷毀,利弊取舍與上文行9的敘述相同。

 

  哥哥們姐姐們,那么如果把配置文件的scope屬性改為"prototype"會發生什么呢?讓我們看一下打印的日志:

1  2015-11-07 22:37:29,280  INFO [main] (XmlBeanDefinitionReader.java:315) - Loading XML bean definitions from class path resource [com/mesopotamia/bean_life_cycle/jayConfig.xml]
2  MyInstantiationAwareBeanPostProcessor.postProcessBeforeInstantiation
3  2015-11-07 22:37:29,407  INFO [main] (JayChou.java:18) - 調用JayChou的無參構造函數。傑倫出道啦。
4  InstantiationAwareBeanPostProcessor.postProcessAfterInstantiation
5  InstantiationAwareBeanPostProcessor.postProcessPropertyValues
6  2015-11-07 22:37:29,444  INFO [main] (MyBeanPostProcessor.java:28) - 配置文件中Jay的女朋友是:蔡依林
7  2015-11-07 22:37:29,444  INFO [main] (MyBeanPostProcessor.java:30) - 調用BeanPostProcessor的postProcessBeforeInitialization處理后,Jay的女朋友變為:侯佩岑
8  2015-11-07 22:37:29,444  INFO [main] (JayChou.java:49) - 調用InitializingBean.afterPropertiesSet(),屬性配置完畢了再做些善后工作。
9  2015-11-07 22:37:29,445  INFO [main] (JayChou.java:61) - 通過調用配置文件初始化女朋友為:徐若瑄
10  2015-11-07 22:37:29,445  INFO [main] (MyBeanPostProcessor.java:16) - JayChou當前的女朋友是:徐若瑄
11  2015-11-07 22:37:29,445  INFO [main] (MyBeanPostProcessor.java:18) - 調用BeanPostProcessor的postProcessAfterInitialization處理后,JayChou的女朋友變成:昆凌
12  2015-11-07 22:37:29,446  INFO [main] (BeanLifeCycleMain.java:26) - jay:JayChou簡介:   性別:男    職業:歌手   女朋友:昆凌
13  2015-11-07 22:37:29,446  INFO [main] (JayChou.java:18) - 調用JayChou的無參構造函數。傑倫出道啦。
14  InstantiationAwareBeanPostProcessor.postProcessAfterInstantiation
15  InstantiationAwareBeanPostProcessor.postProcessPropertyValues
16  2015-11-07 22:37:29,447  INFO [main] (MyBeanPostProcessor.java:28) - 配置文件中Jay的女朋友是:蔡依林
17  2015-11-07 22:37:29,447  INFO [main] (MyBeanPostProcessor.java:30) - 調用BeanPostProcessor的postProcessBeforeInitialization處理后,Jay的女朋友變為:侯佩岑
18  2015-11-07 22:37:29,447  INFO [main] (JayChou.java:49) - 調用InitializingBean.afterPropertiesSet(),屬性配置完畢了再做些善后工作。
19  2015-11-07 22:37:29,447  INFO [main] (JayChou.java:61) - 通過調用配置文件初始化女朋友為:徐若瑄
20  2015-11-07 22:37:29,448  INFO [main] (MyBeanPostProcessor.java:16) - JayChou當前的女朋友是:徐若瑄
21  2015-11-07 22:37:29,448  INFO [main] (MyBeanPostProcessor.java:18) - 調用BeanPostProcessor的postProcessAfterInitialization處理后,JayChou的女朋友變成:昆凌
22  2015-11-07 22:37:29,448  INFO [main] (BeanLifeCycleMain.java:31) - jay2:JayChou簡介:  性別:男    職業:歌手   女朋友:溫嵐
23  2015-11-07 22:37:29,448  INFO [main] (BeanLifeCycleMain.java:32) - jay:JayChou簡介:   性別:男    職業:歌手   女朋友:昆凌
24  2015-11-07 22:37:29,449  INFO [main] (BeanLifeCycleMain.java:34) - false
25  2015-11-07 22:37:29,449  INFO [main] (DefaultSingletonBeanRegistry.java:422) - Destroying singletons in org.springframework.beans.factory.xml.XmlBeanFactory@ce5b1c: defining beans [jay]; root of factory hierarchy

上面的日志是見證奇跡的時刻,當創建jay2對象時,從13行到21行,又進行了一次初始化的過程,而22行到24行發現,兩個對象不相同了,這就是prototype的作用。

 

 ApplicationContext中Bean的生命周期與BeanFactory類似,但是又有不同。對於InstantiationAwareBeanPostProcessor和MyBeanPostProcessor,BeanFactory需要在代碼中注冊方才能使用,而ApplicationContext只需要在xml中配置,spring會自動將它們注冊到應用上下文中,這是二者最大的區別,也是為什么普遍使用ApplicationContext而非BeanFactory的原因。ApplicationContext是BeanFactory的擴展類。

 

  spring當前在各大企業應用中廣受青睞,spring融匯的java思想也堪稱經典,因此筆者后面將繼續跟廣大猿猿一塊學習探討spring的精髓,對於文中的錯誤與不足,抑或是讀者有一般人不告訴他的精辟見解,還望在評論中留言,一起學習。共勉。

 


免責聲明!

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



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