輕鬆學習Spring IoC容器和Dependency Injection模式(轉)

用spring很長時間了,但是一直都是一知半解,今天準備開始學習spring2.5,從基礎開始看,這篇文章寫的還不錯,轉過來作爲保留:)
原文地址:http://www.blogjava.net/rickhunter/articles/29015.html

最近公司需要,項目中要用到SpringIbatis。趁着過年好好學習學習。Ibatis就如同Hibernate一樣的持久層技術,學習起來難度不大,但Spring可不一樣,揣着IocDJAOP,四處走紅。學起來可不容易。就市面上而言,就一本《expert one-on-one J2EE Development without EJB中文版》值得參考,爲了生活,再貴也得買。這本書的前五章都是說EJB的不是,從第六章開始進入正題,介紹控制反轉,以後基本都是說Spring了。

       可能本人比較愚笨,控制反轉弄得不明白。這樣就得上網上找答案了。最後在一個叫Bromonblog上找到的淺顯易懂的答案。下面就是引用他說的話:


IoCDI

  首先想說說IoCInversion of Control,控制倒轉)。這是spring的核心,貫穿始終。所謂IoC,對於spring框架來說,就是由spring來負責控制對象的生命週期和 對象間的關係。這是什麼意思呢,舉個簡單的例子,我們是如何找女朋友的?常見的情況是,我們到處去看哪裏有長得漂亮身材又好的mm,然後打聽她們的興趣愛 好、qq號、電話號、ip號、iq………,想辦法認識她們,投其所好送其所要,然後嘿嘿……這個過程是複雜深奧的,我們必須自己設計和麪對每個環節。傳 統的程序開發也是如此,在一個對象中,如果要使用另外的對象,就必須得到它(自己new一個,或者從JNDI中查詢一個),使用完之後還要將對象銷燬(比 Connection等),對象始終會和其他的接口或類藕合起來。

  那麼IoC是如何做的呢?有點像通過婚介找女朋友,在我和女朋友之間引入了一個第三者:婚姻介紹所。婚介管理了很多男男女女的資料,我可以向婚 介提出一個列表,告訴它我想找個什麼樣的女朋友,比如長得像李嘉欣,身材像林熙雷,唱歌像周杰倫,速度像卡洛斯,技術像齊達內之類的,然後婚介就會按照我 們的要求,提供一個mm,我們只需要去和她談戀愛、結婚就行了。簡單明瞭,如果婚介給我們的人選不符合要求,我們就會拋出異常。整個過程不再由我自己控 制,而是有婚介這樣一個類似容器的機構來控制。Spring所倡導的開發方式就是如此,所有的類都會在spring容器中登記,告訴spring你是個什 麼東西,你需要什麼東西,然後spring會在系統運行到適當的時候,把你要的東西主動給你,同時也把你交給其他需要你的東西。所有的類的創建、銷燬都由 spring來控制,也就是說控制對象生存週期的不再是引用它的對象,而是spring。對於某個具體的對象而言,以前是它控制其他對象,現在是所有對象 都被spring控制,所以這叫控制反轉。如果你還不明白的話,我決定放棄。

IoC的一個重點是在系統運行中,動態的向某個對象提供它所需要的其他對象。這一點是通過DIDependency Injection,依賴注入)來實現的。比如對象A需要操作數據庫,以前我們總是要在A中自己編寫代碼來獲得一個Connection對象,有了 spring我們就只需要告訴springA中需要一個Connection,至於這個Connection怎麼構造,何時構造,A不需要知道。在系統 運行時,spring會在適當的時候製造一個Connection,然後像打針一樣,注射到A當中,這樣就完成了對各個對象之間關係的控制。A需要依賴 Connection才能正常運行,而這個Connection是由spring注入到A中的,依賴注入的名字就這麼來的。那麼DI是如何實現的呢? Java 1.3之後一個重要特徵是反射(reflection),它允許程序在運行的時候動態的生成對象、執行對象的方法、改變對象的屬性,spring就是通過反射來實現注入的。關於反射的相關資料請查閱java doc


    我想通過Bromon的介紹,大家對IoCDI都有了比較生動的理解了。再來看看《expert one-on-one J2EE Development without EJB中文版》是怎麼解釋這兩個概念的。書上是這麼說的:

IoC是一個很大的概念,可以用不同的方式來實現。主要的實現形式有兩種:

依賴查找:容器提供回調接口和上下文環境給組件。EJBApache Avalon都是使用這種方式。

依賴注入:組件不做定位查詢,只是提供普通的Java方法讓容器去決定依賴關係。容器全權負責組件的裝配,它會把符合依賴關係的對象通過JavaBean屬性或者構造子傳遞給需要的對象。通過JavaBean屬性注射依賴關係的做法稱爲設值方法注入(Setter Injection);將依賴關係作爲構造子參數傳入的做法稱爲構造子注入(Constructor Injection)。

附圖說明:


spring_2.jpg

   

到這裏,大家應該對IoCDI都有了初步的認識了。其實就Spring來說,就是JavaBeanSpring來管理組裝,表面上看就少了幾個new字,其實就是爲了降低耦合度,這也是我們做軟件的目標之一。

至於Spring是怎樣實現IoC的,expert one-on-one J2EE Development without EJB中文版》第七章“Spring框架介紹”很詳細的列舉了多種方法。說實在,一下子看這麼多,我真有點糊塗了。我還是自己寫個Demo熟悉一下大名鼎鼎的Spring吧。

首先得下載SpringSpring網上有兩種Spring 包一種是spring-framework-1.2.6-with-dependencies.zip,另一種是spring-framework-1.2.6.zip。當然最好是下載spring-framework-1.2.6-with-dependencies.zip形式的,因爲裏面包括了更多的東東。spring-framework-1.2.6-with-dependencies.zip的下載地址是:http://prdownloads.sourceforge.net/springframework/spring-framework-1.2.6-with-dependencies.zip

下載下來,解壓後,你會發現裏面有很多jar文件。因爲剛剛接觸Spring,因此我只需要spring-core.jarspring-framework-1.2.6/dist),將其導入eclipse的構建路徑中。另外,log日誌是需要的,這也是爲了養成良好的編程習慣。將log4j-1.2.9.jarspring-framework-1.2.6/lib/log4j)和commons-logging.jarspring-framework-1.2.6/lib/jakarta-commons)導入到構建路徑中。

準備就緒,開始寫Demo了。

我的工程的結構是:

   spring_1.jpg

1、 log4j.properties代碼:

log4j.rootLogger=Debug, stdout
log4j.appender.stdout
=org.apache.log4j.ConsoleAppender
log4j.appender.stdout.layout
=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern
=%c{1- %m%n

如何使用Log4j,請看我的另一篇轉貼的文章:如何使用Log4J

2、 HelloBean的代碼:

package com;

public class HelloBean {
    
private String helloworld="Hello!World!";
    
    
public String getHelloworld() {
        
return helloworld;
    }
    
    
public void setHelloworld(String helloworld) {
        
this.helloworld = helloworld;
    }
}

這是一個簡單的JavaBean,有個String類型的helloworld屬性,初始值是"Hello!World!" 

3、 Bean.xml代碼:

<?xml version="1.0" encoding="GBK"?>
<!DOCTYPE beans PUBLIC "-//SPRING/DTD BEAN/EN" 
    
"http://www.springframework.org/dtd/spring-beans.dtd">

<beans>
   
<bean id="helloBean" class="com.HelloBean">
        
<property name="helloworld">
            
<value>Hello!Rick</value>
        
</property>
   
</bean>
</beans>

        Spirng重點之一就是配置文件,上面是個相當簡單的配置文件,我想大家都應該看得懂。最後就是寫應用程序了。

4、 Test的代碼:

package com;

import org.springframework.beans.factory.*;
import org.springframework.beans.factory.xml.XmlBeanFactory;
import org.springframework.core.io.ClassPathResource;
import org.springframework.core.io.Resource;

public class Test {

    
public static void main(String[] args) {
        
//實例化JavaBean,主要是爲了比較new對象和依賴注入兩者的區別
        HelloBean hellobean=new HelloBean();
        System.out.println(hellobean.getHelloworld());
        
        
//通過Spring訪問JavaBean組件
        Resource resource=new ClassPathResource("com/bean.xml");
        BeanFactory factory
=new XmlBeanFactory(resource);
        hellobean
=(HelloBean)factory.getBean("helloBean");
        System.out.println(hellobean.getHelloworld());
    }
}

    這個Demo很好的闡述了Spring的Ioc,其實就Spring而言,就是通過配置文件,讓Spring如同一個管家一樣來管理所有的Bean類。

    Spring的依賴注入相對複雜一點,主要是明白調用別的Bean,不是通過實例化對象來調用,而是告訴Spring,我需要什麼Bean,然後Spring再向你的Bean裏面注入你所需要的Bean對象。
    接下來說說代碼實現,我只是在剛剛的例子上再添加一點東東。
    首先要增加一個HelloImp的接口,這是問什麼呢,那你得問Spring,它定的規矩:JavaBean的實現要有兩個部分,一個接口,一個默認實現。你不照做就不行。
    HelloImp代碼:  
     
package com;

public interface HelloImp {
    
public void getName();
}

   
    實現HelloImp的Hello代碼:
  
package com;

public class Hello implements HelloImp {
    
public void getName(){
        System.out.println(
"Jack");
    }
}
         
    接着就是在HelloBean中調用Hello了。Spring的不同之處也在這體現出來。

package com;

public class HelloBean {
    
private String helloworld="Hello!World!";
    
private HelloImp hello;  //注意這個私有對象是藉口

    
public String getHelloworld() {
        
return helloworld;
    }
    
    
public void setHelloworld(String helloworld) {
        
this.helloworld = helloworld;
    }
    
    
public void setHello(HelloImp hello) {
        
this.hello = hello;
    }
    
  
public void get(){
        
this
.hello.getName();
   }

}
    注意字體加粗的地方。


    配置文件也需要增加一點東西:


   
<?xml version="1.0" encoding="GBK"?>
<!DOCTYPE beans PUBLIC "-//SPRING/DTD BEAN/EN" 
    
"http://www.springframework.org/dtd/spring-beans.dtd">

<beans>
<!—注意引用的類是具體的類Hello-->
   
<bean id="myHello" class="com.Hello">
   
</bean>
   
<bean id="helloBean" class="com.HelloBean">
        
<property name="helloworld">
            
<value>Hello!Rick</value>
        
</property>
       
<property name="hello">
           
<ref bean="myHello"></ref>
       
</property>

   
</bean>
</beans>

    注意字體加粗的部分。

    最後在Test中添加一句hellobean.get();就可以看到結果了。

package com;

import org.springframework.beans.factory.*;
import org.springframework.beans.factory.xml.XmlBeanFactory;
import org.springframework.core.io.ClassPathResource;
import org.springframework.core.io.Resource;

public class Test {

    
public static void main(String[] args) {
        HelloBean hellobean
=new HelloBean();
        System.out.println(hellobean.getHelloworld());
        
        Resource resource
=new ClassPathResource("com/bean.xml");
        BeanFactory factory
=new XmlBeanFactory(resource);
        
        hellobean
=(HelloBean)factory.getBean("helloBean");
        System.out.println(hellobean.getHelloworld());
        hellobean.get();
    }
}


到這,Spring的IoC和DI總算有了一定的認識,也算是敲開了Spring的大門了。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章