spring學習——Ioc基礎三(Ioc配置使用)

1、XML配置的結構 java

通常配置文件結構以下:spring

<beans>  
    <import resource=」resource1.xml」/>  
    <bean id=」bean1」class=」」></bean>  
    <bean id=」bean2」class=」」></bean>  
<bean name=」bean2」class=」」></bean>  
    <alias alias="bean3" name="bean2"/>  
    <import resource=」resource2.xml」/>  
</beans>

一、<bean>標籤主要用來進行Bean定義;數組

二、alias用於定義Bean別名的;ide

三、import用於導入其餘配置文件的Bean定義,這是爲了加載多個配置文件,固然也能夠把這些配置文件構造爲一個數組(new String[] {「config1.xml」, config2.xml})傳給ApplicationContext實現進行加載多個配置文件,那一個更適合由用戶決定;這兩種方式都是經過調用Bean Definition Reader 讀取Bean定義,內部實現沒有任何區別。<import>標籤能夠放在<beans>下的任何位置,沒有順序關係。測試


2、Bean的配置ui

Spring IoC容器目的就是管理Bean,這些Bean將根據配置文件中的Bean定義進行建立,而Bean定義在容器內部由BeanDefinition對象表示,該定義主要包含如下信息:this

●全限定類名(FQN):用於定義Bean的實現類;spa

●Bean行爲定義:這些定義了Bean在容器中的行爲;包括做用域(單例、原型建立)、是否惰性初始化及生命週期等;code

●Bean建立方式定義:說明是經過構造器仍是工廠方法建立Bean;orm

●Bean之間關係定義:即對其餘bean的引用,也就是依賴關係定義,這些引用bean也能夠稱之爲同事bean 或依賴bean,也就是依賴注入。

Bean定義只有「全限定類名」在當使用構造器或靜態工廠方法進行實例化bean時是必須的,其餘都是可選的定義。難道Spring只能經過配置方式來建立Bean嗎?回答固然不是,某些SingletonBeanRegistry接口實現類實現也容許將那些非BeanFactory建立的、已有的用戶對象註冊到容器中,這些對象必須是共享的,好比使用DefaultListableBeanFactory 的registerSingleton() 方法。不過建議採用元數據定義。


3、Bean的命名

      每一個Bean能夠有一個或多個id(或稱之爲標識符或名字),在這裏咱們把第一個id稱爲「標識符」,其他id叫作「別名」;這些id在IoC容器中必須惟一。如何爲Bean指定id呢,有如下幾種方式;


一、不指定id,只配置必須的全限定類名,由IoC容器爲其生成一個標識,客戶端必須經過接口「T getBean(Class<T> requiredType)」獲取Bean;

<bean class="cn.javass.spring.chapter2.helloworld.HelloImpl"/>

測試代碼片斷以下:

@Test  
public void test1() {  
BeanFactory beanFactory =  
   new ClassPathXmlApplicationContext("chapter2/namingbean1.xml");  
    //根據類型獲取bean  
    HelloApi helloApi = beanFactory.getBean(HelloApi.class);  
    helloApi.sayHello();  
}


二、指定id,必須在Ioc容器中惟一;

<bean id=」 bean」 class=」 cn.javass.spring.chapter2.helloworld.HelloImpl」/>    (2)

測試代碼片斷以下:

@Test  
public void test2() {  
BeanFactory beanFactory = new ClassPathXmlApplicationContext("chapter2/namingbean2.xml");  
//根據id獲取bean  
    HelloApi bean = beanFactory.getBean("bean", HelloApi.class);  
    bean.sayHello();  
}


三、指定name,這樣name就是「標識符」,必須在Ioc容器中惟一;

<bean name=」 bean」 class =」 cn.javass.spring.chapter2.helloworld.HelloImpl」/>

測試代碼片斷以下:

@Test  
public void test3() {  
    BeanFactory beanFactory =  
new ClassPathXmlApplicationContext("chapter2/namingbean3.xml");  
    //根據name獲取bean  
HelloApi bean = beanFactory.getBean("bean", HelloApi.class);  
bean.sayHello();  
}


四、指定id和name,id就是標識符,而name就是別名,必須在Ioc容器中惟一;

<bean id=」bean1」 name=」alias1」 class=」cn.javass.spring.chapter2.helloworld.HelloImpl」/>  
    <!-- 若是id和name同樣,IoC容器能檢測到,並消除衝突 -->  
<bean id="bean3" name="bean3" class="cn.javass.spring.chapter2.helloworld.HelloImpl"/>

測試代碼片斷以下:

@Test  
public void test4() {  
BeanFactory beanFactory = new ClassPathXmlApplicationContext("chapter2/namingbean4.xml");  
    //根據id獲取bean  
    HelloApi bean1 = beanFactory.getBean("bean1", HelloApi.class);  
    bean1.sayHello();  
    //根據別名獲取bean  
    HelloApi bean2 = beanFactory.getBean("alias1", HelloApi.class);  
    bean2.sayHello();  
    //根據id獲取bean  
    HelloApi bean3 = beanFactory.getBean("bean3", HelloApi.class);  
    bean3.sayHello();  
    String[] bean3Alias = beanFactory.getAliases("bean3");  
    //所以別名不能和id同樣,若是同樣則由IoC容器負責消除衝突  
    Assert.assertEquals(0, bean3Alias.length);  
}


5、指定多個name,多個name用「,」、「;」、「 」分割,第一個被用做標識符,其餘的(alias一、alias二、alias3)是別名,全部標識符也必須在Ioc容器中惟一;

<bean name=」 bean1;alias11,alias12;alias13 alias14」 class=」 cn.javass.spring.chapter2.helloworld.HelloImpl」/>     
<!-- 當指定id時,name指定的標識符所有爲別名 -->  
<bean id="bean2" name="alias21;alias22"  
class="cn.javass.spring.chapter2.helloworld.HelloImpl"/>

測試代碼片斷以下:

@Test  
public void test5() {  
BeanFactory beanFactory =  
new ClassPathXmlApplicationContext("chapter2/namingbean5.xml");  
    //1根據id獲取bean  
    HelloApi bean1 = beanFactory.getBean("bean1", HelloApi.class);  
    bean1.sayHello();  
    //2根據別名獲取bean  
    HelloApi alias11 = beanFactory.getBean("alias11", HelloApi.class);  
    alias11.sayHello();  
    //3驗證確實是四個別名         
    String[] bean1Alias = beanFactory.getAliases("bean1");  
    System.out.println("=======namingbean5.xml bean1 別名========");  
    for(String alias : bean1Alias) {  
        System.out.println(alias);  
    }  
    Assert.assertEquals(4, bean1Alias.length);  
    //根據id獲取bean  
    HelloApi bean2 = beanFactory.getBean("bean2", HelloApi.class);  
    bean2.sayHello();  
    //2根據別名獲取bean  
    HelloApi alias21 = beanFactory.getBean("alias21", HelloApi.class);  
    alias21.sayHello();  
    //驗證確實是兩個別名  
    String[] bean2Alias = beanFactory.getAliases("bean2");  
    System.out.println("=======namingbean5.xml bean2 別名========");  
    for(String alias : bean2Alias) {  
        System.out.println(alias);  
    }     
    Assert.assertEquals(2, bean2Alias.length);     
}


六、使用<alias>標籤指定別名,別名也必須在IoC容器中惟一


<bean name="bean" class="cn.javass.spring.chapter2.helloworld.HelloImpl"/>  
<alias alias="alias1" name="bean"/>  
<alias alias="alias2" name="bean"/>

測試代碼片斷以下:

@Test  
public void test6() {  
BeanFactory beanFactory =  
new ClassPathXmlApplicationContext("chapter2/namingbean6.xml");  
    //根據id獲取bean  
    HelloApi bean = beanFactory.getBean("bean", HelloApi.class);  
   bean.sayHello();  
    //根據別名獲取bean  
    HelloApi alias1 = beanFactory.getBean("alias1", HelloApi.class);  
    alias1.sayHello();  
    HelloApi alias2 = beanFactory.getBean("alias2", HelloApi.class);  
    alias2.sayHello();  
    String[] beanAlias = beanFactory.getAliases("bean");  
    System.out.println("=======namingbean6.xml bean 別名========");  
    for(String alias : beanAlias) {  
        System.out.println(alias);  
   }  
   System.out.println("=======namingbean6.xml bean 別名========");  
    Assert.assertEquals(2, beanAlias.length);  
 }

以上測試代碼在cn.javass.spring.chapter2.NamingBeanTest.java文件中。


從定義來看,name或id若是指定它們中的一個時都做爲「標識符」,那爲何還要有id和name同時存在呢?這是由於當使用基於XML的配置元數據時,在XML中id是一個真正的XML id屬性,所以當其餘的定義來引用這個id時就體現出id的好處了,能夠利用XML解析器來驗證引用的這個id是否存在,從而更早的發現是否引用了一個不存在的bean,而使用name,則可能要在真正使用bean時才能發現引用一個不存在的bean。

Bean命名約定:Bean的命名遵循XML命名規範,但最好符合Java命名規範,由「字母、數字、下劃線組成「,並且應該養成一個良好的命名習慣, 好比採用「駝峯式」,即第一個單詞首字母開始,從第二個單詞開始首字母大寫開始,這樣能夠增長可讀性。


4、實例化Bean


Spring IoC容器如何實例化Bean呢?傳統應用程序能夠經過new和反射方式進行實例化Bean。而Spring IoC容器則須要根據Bean定義裏的配置元數據使用反射機制來建立Bean。在Spring IoC容器中根據Bean定義建立Bean主要有如下幾種方式:


一、使用構造器實例化Bean:這是最簡單的方式,Spring IoC容器即能使用默認空構造器也能使用有參數構造器兩種方式建立Bean,如如下方式指定要建立的Bean類型:

 使用空構造器進行定義,使用此種方式,class屬性指定的類必須有空構造器

<bean name="bean1" class="cn.javass.spring.chapter2.HelloImpl2"/>

使用有參數構造器進行定義,使用此中方式,可使用< constructor-arg >標籤指定構造器參數值,其中index表示位置,value表示常量值,也能夠指定引用,指定引用使用ref來引用另外一個Bean定義,後邊會詳細介紹:

<bean name="bean2" class="cn.javass.spring.chapter2.HelloImpl2">  
    <!-- 指定構造器參數 -->  
    <constructor-arg index="0" value="Hello Spring!"/>  
</bean>


知道如何配置了,讓咱們作個例子的例子來實踐一下吧:

(1)準備Bean class(HelloImpl2.java),該類有一個空構造器和一個有參構造器:


package cn.javass.spring.chapter2;   
  public class HelloImpl2 implements HelloApi {  
           private String message;  
          public HelloImpl2() {  
                  this.message = "Hello World!";  
           }  
         Public HelloImpl2(String message) {  
                  this.message = message;  
           }  
           @Override  
           public void sayHello() {  
                  System.out.println(message);  
           }  
}

(2)在配置文件(resources/chapter2/instantiatingBean.xml)配置Bean定義,以下所示:


<!--使用默認構造參數-->  
<bean name="bean1" class="cn.javass.spring.chapter2.HelloImpl2"/>  
    <!--使用有參數構造參數-->  
<bean name="bean2" class="cn.javass.spring.chapter2.HelloImpl2">  
<!-- 指定構造器參數 -->  
    <constructor-arg index="0" value="Hello Spring!"/>  
</bean>

 (3)配置完了,讓咱們寫段測試代碼(InstantiatingContainerTest)來看下是否工做吧:

@Test  
public void testInstantiatingBeanByConstructor() {  
       //使用構造器  
     BeanFactory beanFactory =  
new ClassPathXmlApplicationContext("chapter2/instantiatingBean.xml");  
       HelloApi bean1 = beanFactory.getBean("bean1", HelloApi.class);  
       bean1.sayHello();  
       HelloApi bean2 = beanFactory.getBean("bean2", HelloApi.class);  
       bean2.sayHello();  
}

二、使用靜態工廠方式實例化Bean,使用這種方式除了指定必須的class屬性,還要指定factory-method屬性來指定實例化Bean的方法,並且使用靜態工廠方法也容許指定方法參數,spring IoC容器將調用此屬性指定的方法來獲取Bean,配置以下所示:


(1)先來看看靜態工廠類代碼吧HelloApiStaticFactory:

public class HelloApiStaticFactory {  
       //工廠方法  
       public static HelloApi newInstance(String message) {  
           //返回須要的Bean實例  
           return new HelloImpl2(message);  
    }  
}

(2)靜態工廠寫完了,讓咱們在配置文件(resources/chapter2/instantiatingBean.xml)配置Bean定義:

 java代碼:

<!-- 使用靜態工廠方法 -->  
<bean id="bean3" class="cn.javass.spring.chapter2.HelloApiStaticFactory" factory-method="newInstance">  
     <constructor-arg index="0" value="Hello Spring!"/>  
</bean>

(3)配置完了,寫段測試代碼來測試一下吧,InstantiatingBeanTest:

 java代碼:

@Test  
public void testInstantiatingBeanByStaticFactory() {  
       //使用靜態工廠方法  
       BeanFactory beanFactory =  
new ClassPathXmlApplicationContext("chaper2/instantiatingBean.xml");  
       HelloApi bean3 = beanFactory.getBean("bean3", HelloApi.class);  
       bean3.sayHello();  
}

 

三、使用實例工廠方法實例化Bean,使用這種方式不能指定class屬性,此時必須使用factory-bean屬性來指定工廠Bean,factory-method屬性指定實例化Bean的方法,並且使用實例工廠方法容許指定方法參數,方式和使用構造器方式同樣,配置以下:

(1)實例工廠類代碼(HelloApiInstanceFactory.java)以下:

java代碼:   

package cn.javass.spring.chapter2;  
public class HelloApiInstanceFactory {  
public HelloApi newInstance(String message) {  
          return new HelloImpl2(message);  
   }  
}

   (2)讓咱們在配置文件(resources/chapter2/instantiatingBean.xml)配置Bean定義:

 java代碼:

<!—一、定義實例工廠Bean -->  
<bean id="beanInstanceFactory" class="cn.javass.spring.chapter2.HelloApiInstanceFactory"/>  
<!—二、使用實例工廠Bean建立Bean -->  
<bean id="bean4" factory-bean="beanInstanceFactory"  factory-method="newInstance">  
 <constructor-arg index="0" value="Hello Spring!"></constructor-arg>  
</bean>

(3)測試代碼InstantiatingBeanTest:

 java代碼:

@Test  
public void testInstantiatingBeanByInstanceFactory() {  
//使用實例工廠方法  
       BeanFactory beanFactory =  
            new ClassPathXmlApplicationContext("chapter2/instantiatingBean.xml");  
       HelloApi bean4 = beanFactory.getBean("bean4", HelloApi.class);  
       bean4.sayHello();  
}

       經過以上例子咱們已經基本掌握瞭如何實例化Bean了,你們是否注意到?這三種方式只是配置不同,從獲取方式看徹底同樣,沒有任何不一樣。這也是Spring IoC的魅力,Spring IoC幫你建立Bean,咱們只管使用就能夠了,是否是很簡單。


小結:

       到此咱們已經講完了Spring IoC基礎部分,包括IoC容器概念,如何實例化容器,Bean配置、命名及實例化,Bean獲取等等。不知你們是否注意到到目前爲止,咱們只能經過簡單的實例化Bean,沒有涉及Bean之間關係。接下來一章讓咱們進入配置Bean之間關係章節,也就是依賴注入。

相關文章
相關標籤/搜索