Spring BeanFactory例項化Bean的詳細過程

木叔發表於2017-02-17
Spring中Bean的例項化是Bean生命週期的一個重要環節,通常Bean初始化後將不再改變。
那麼Spring例項Bean的過程到底是怎麼樣的呢?!
 
要想獲取到一個bean物件,得先通過BeanFactory的getBean()方法獲取,期間會經過一系列步驟來例項化這個bean物件:
第一步:呼叫Bean的預設構造方法(當然也可以是指定的其它構造方法),生成bean例項:bean1。
第二步:檢查Bean配置檔案中是否注入了Bean的屬性值,如果有注入,則在bean1例項的基礎上對其屬性進行注入,把原來的bean1給覆蓋掉形成新的bean例項:bean2。
第三步:檢查Bean是否實現了InitializingBean介面,如果實現了此介面,則呼叫afterPropertiesSet()方法對bean2進行相應操作後,把bean2覆蓋形成新的bean例項:bean3。
第四步:檢查Bean配置檔案中是否指定了init-method此屬性,如果已指定,則呼叫此屬性對應方法並對bean3進行相應操作後,最終把bean3覆蓋形成新的例項:bean4。
通過上面的步驟我們發現,Spring例項一個bean時,這個bean是在不斷的變化的!
 
為了更好的說明以上步驟,請看下面程式碼:
實體類:
/** 
 * 實體類 
 */  
public class Employee implements InitializingBean, DisposableBean, BeanNameAware {  
    private String id;// 員工編號  
    private String name;// 員工姓名  
    private String sex;// 員工性別  
    private String age;// 員工年齡  
    private String nativePlace;// 員工籍貫  
    private String department;// 員工部門  
    private String beanName;// bean的名稱  
  
    public Employee() {  
        System.out.println("**********第一步:呼叫Bean的預設構造方法**********");  
        this.id = "bean1:G080405214";  
        System.out.println("bean1的 值:" + this);  
        System.out.println("**********第二步:檢查Bean配置檔案中是否注入了Bean的屬性值**********");  
    }  
  
    public void afterPropertiesSet() throws Exception {  
        System.out.println("bean2的值:" + this);  
        System.out.println("**********第三步:檢查Bean是否實現了InitializingBean介面**********");  
        this.name = "bean3:李曉紅";  
        this.sex = "bean3:女";  
        this.age = "bean3:25";  
        System.out.println("bean3的值:" + this);  
    }  
  
    public void init() {  
        System.out  
                .println("**********第四步:檢查Bean配置檔案中是否指定了init-method此屬性**********");  
        this.nativePlace = "bean3:北京";  
        System.out.println("bean4的值:" + this);  
    }  
  
    public void destroy() throws Exception {  
        System.out.println("**********服務停止**********");  
    }  
  
    public void setBeanName(String arg0) {  
        System.out.println("**********設定bean的名稱**********");  
        this.beanName = "myBeanName";  
  
    }  
  
    public String getId() {  
        return id;  
    }    
    public void setId(String id) {  
        this.id = id;  
    }    
    public String getName() {  
        return name;  
    }    
    public void setName(String name) {  
        this.name = name;  
    }   
    public String getSex() {  
        return sex;  
    }    
    public void setSex(String sex) {  
        this.sex = sex;  
    }  
  
    public String getAge() {  
        return age;  
    }  
  
    public void setAge(String age) {  
        this.age = age;  
    }  
  
    public String getNativePlace() {  
        return nativePlace;  
    }  
  
    public void setNativePlace(String nativePlace) {  
        this.nativePlace = nativePlace;  
    }  
  
    public String getDepartment() {  
        return department;  
    }  
  
    public void setDepartment(String department) {  
        this.department = department;  
    }  
  
    public String getBeanName() {  
        return beanName;  
    }  
  
    @Override  
    public String toString() {  
        return "Employee [id=" + id + ", name=" + name + ", sex=" + sex  
                + ", age=" + age + ", nativePlace=" + nativePlace  
                + ", department=" + department + ", beanName=" + beanName + "]";  
    }  
}  

 工具類:

/** 
 * Bean上下文工具類 
 */  
public class BeanContextHelper {  
    private static ApplicationContext _instance;  
  
    static {  
        if (_instance == null)  
            _instance = buildApplicationContext();  
    }  
  
    private BeanContextHelper() {  
    }  
  
    /** 
     * 重新構建ApplicationContext物件  
     */  
    public static ApplicationContext buildApplicationContext() {  
        return new ClassPathXmlApplicationContext("applicationContext-base.xml");  
    }  
  
    /** 
     * 獲取一個ApplicationContext物件 
     */  
    public static ApplicationContext getApplicationContext() {  
        return _instance;  
    }  
}  

 Spring的Bean配置:

<?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:context="http://www.springframework.org/schema/context"  
    xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"  
    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-2.5.xsd  
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd  
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">  
  
    <!--==============測試Spring BeanFactory例項化Bean的過程-->  
    <bean id="employee" class="bean_factory_test.Employee"  
        init-method="init" destroy-method="destroy">  
        <!--預設部門為研發部的-->  
        <property name="department">  
            <value>bean2:研發部</value>  
        </property>  
    </bean>  
  
</beans> 

 測試類:

/** 
 * BeanFactory例項化Bean工程測試類  
 */  
public class Test {  
    public static void main(String args[]) {  
        Test test = new Test();  
        test.test();  
    }  
  
    public void test() {  
        ApplicationContext context = BeanContextHelper.getApplicationContext();  
        Employee employee = (Employee) context.getBean("employee");  
        System.out.println("**********從Spring BeanFactory獲取到的最終bean例項**********");  
        System.out.println("最終bean的值:" + employee);  
    }  
} 

 執行結果:

**********第一步:呼叫Bean的預設構造方法**********  
bean1的 值:Employee [id=bean1:G080405214, name=null, sex=null, age=null, nativePlace=null, department=null, beanName=null]  
**********第二步:檢查Bean配置檔案中是否注入了Bean的屬性值**********  
**********設定bean的名稱**********  
bean2的值:Employee [id=bean1:G080405214, name=null, sex=null, age=null, nativePlace=null, department=bean2:研發部, beanName=myBeanName]  
**********第三步:檢查Bean是否實現了InitializingBean介面**********  
bean3的值:Employee [id=bean1:G080405214, name=bean3:李曉紅, sex=bean3:女, age=bean3:25, nativePlace=null, department=bean2:研發部, beanName=myBeanName]  
**********第四步:檢查Bean配置檔案中是否指定了init-method此屬性**********  
bean4的值:Employee [id=bean1:G080405214, name=bean3:李曉紅, sex=bean3:女, age=bean3:25, nativePlace=bean3:北京, department=bean2:研發部, beanName=myBeanName]  
**********從Spring BeanFactory獲取到的最終bean例項**********  
最終bean的值:Employee [id=bean1:G080405214, name=bean3:李曉紅, sex=bean3:女, age=bean3:25, nativePlace=bean3:北京, department=bean2:研發部, beanName=myBeanName]  

 從執行結果看,我們應該很清楚Bean例項化的具體過程了。

Employee實現了3個介面:
InitializingBean:此介面提供afterPropertiesSet()方法,它的作用是為bean提供了定義初始化的功能。
DisposableBean:此介面提供destroy()方法,它的作用是在bean例項銷燬前提供操作的功能。
BeanNameAware:此介面提供setBeanName()方法,它的作用是提供設定bean名稱的功能,從上面的執行結果可以看出,此方法是在第二步進行的。
 
 
 

相關文章