1. 程式人生 > >spring的五種依賴注入方式

spring的五種依賴注入方式

平常的java開發中,程式設計師在某個類中需要依賴其它類的方法,則通常是new一個依賴類再呼叫類例項的方法,這種開發存在的問題是new的類例項不好統一管理,spring提出了依賴注入的思想,即依賴類不由程式設計師例項化,而是通過spring容器幫我們new指定例項並且將例項注入到需要該物件的類中。依賴注入的另一種說法是“控制反轉”,通俗的理解是:平常我們new一個例項,這個例項的控制權是我們程式設計師,而控制反轉是指new例項工作不由我們程式設計師來做而是交給spring容器來做。

spring有多種依賴注入的形式,下面介紹spring進行DI的方式:

一.目前使用最廣泛的 @Autowired:自動裝配

自動裝配,用於替代基於XML配置的自動裝配,詳見【3.3.3  自動裝配】。

基於@Autowired的自動裝配,預設是根據型別注入,可以用於構造器、欄位、方法注入,使用方式如下:

java程式碼:

  1. @Autowired(required=true)  
  2. 構造器、欄位、方法  

@Autowired預設是根據引數型別進行自動裝配,且必須有一個Bean候選者注入預設required=true如果允許出現0個Bean候選者需要設定屬性“required=false”,“required”屬性含義和@Required一樣,只是@Required只適用於基於XML配置的setter注入方式,只能打在setting方法上。

(1)、構造器注入:通過將@Autowired註解放在構造器上來完成構造器注入,預設構造器引數通過型別自動裝配,如下所示:

1、準備測試Bean,在構造器上新增@AutoWired註解:

java程式碼:

package cn.javass.spring.chapter12;  
import org.springframework.beans.factory.annotation.Autowired;  
public class TestBean11 {  
    private String message;  
    @Autowired //構造器注入  
    private TestBean11(String message) {  
        this.message = message;  
    }  
    //省略message的getter和setter  
}  

2、在Spring配置檔案(chapter12/dependecyInjectWithAnnotation.xml)新增如下Bean配置:

xml程式碼:

<bean id="testBean11" class="cn.javass.spring.chapter12.TestBean11"/>  

3、測試類如下:

java程式碼:

@Test  
public void testAutowiredForConstructor() {  
    TestBean11 testBean11 = ctx.getBean("testBean11", TestBean11.class);  
    Assert.assertEquals("hello", testBean11.getMessage());  
}  

    在Spring配置檔案中沒有對“testBean11”進行構造器注入和setter注入配置,而是通過在構造器上新增@ Autowired來完成根據引數型別完成構造器注入。

(2)、欄位注入:通過將@Autowired註解放在構造器上來完成欄位注入。

1、準備測試Bean,在欄位上新增@AutoWired註解:

java程式碼:

package cn.javass.spring.chapter12;  
import org.springframework.beans.factory.annotation.Autowired;  
public class TestBean12 {  
    @Autowired //欄位注入  
    private String message;  
    //省略getter和setter  
} 

2、在Spring配置檔案(chapter12/dependecyInjectWithAnnotation.xml)新增如下Bean配置:

xml程式碼:

<bean id="testBean12" class="cn.javass.spring.chapter12.TestBean12"/>  

3、測試方法如下:

java程式碼:

@Test  
public void testAutowiredForField() {  
    TestBean12 testBean12 = ctx.getBean("testBean12", TestBean12.class);  
    Assert.assertEquals("hello", testBean12.getMessage());  
}  

    欄位注入在基於XML配置中無相應概念,欄位注入不支援靜態型別欄位的注入。

(3)、方法引數注入:通過將@Autowired註解放在方法上來完成方法引數注入。

1、準備測試Bean,在方法上新增@AutoWired註解:

java程式碼:

package cn.javass.spring.chapter12;  
import org.springframework.beans.factory.annotation.Autowired;  
public class TestBean13 {  
    private String message;  
    @Autowired //setter方法注入  
    public void setMessage(String message) {  
        this.message = message;  
    }  
    public String getMessage() {  
        return message;  
    }  
}  

java程式碼:

package cn.javass.spring.chapter12;  
//省略import  
public class TestBean14 {  
    private String message;  
    private List<String> list;  
    @Autowired(required = true) //任意一個或多個引數方法注入  
    private void initMessage(String message, ArrayList<String> list) {  
        this.message = message;  
        this.list = list;  
    }  
    //省略getter和setter  
}  

2、在Spring配置檔案(chapter12/dependecyInjectWithAnnotation.xml)新增如下Bean配置:

xml程式碼:

<bean id="testBean13" class="cn.javass.spring.chapter12.TestBean13"/>  
<bean id="testBean14" class="cn.javass.spring.chapter12.TestBean14"/>  
<bean id="list" class="java.util.ArrayList">  
    <constructor-arg index="0">  
        <list>  
            <ref bean="message"/>  
            <ref bean="message"/>  
        </list>  
   </constructor-arg>          
</bean>  

程式碼:

3、測試方法如下:

java程式碼:

@Test  
public void testAutowiredForMethod() {  
    TestBean13 testBean13 = ctx.getBean("testBean13", TestBean13.class);  
    Assert.assertEquals("hello", testBean13.getMessage());  
   
    TestBean14 testBean14 = ctx.getBean("testBean14", TestBean14.class);  
    Assert.assertEquals("hello", testBean14.getMessage());  
    Assert.assertEquals(ctx.getBean("list", List.class), testBean14.getList());  
}  

方法引數注入除了支援setter方法注入,還支援1個或多個引數的普通方法注入,在基於XML配置中不支援1個或多個引數的普通方法注入,方法注入不支援靜態型別方法的注入。

  • 二.setter 方法注入

這是最簡單的注入方式,假設有一個SpringAction,類中需要例項化一個SpringDao物件,那麼就可以定義一個private的SpringDao成員變數,然後建立SpringDao的set方法(這是ioc的注入入口):

Java程式碼  

package com.bless.springdemo.action;  
public class SpringAction {  
        //注入物件springDao  
    private SpringDao springDao;  
        //一定要寫被注入物件的set方法  
        public void setSpringDao(SpringDao springDao) {  
        this.springDao = springDao;  
    }  
  
        public void ok(){  
        springDao.ok();  
    }  
}  

隨後編寫spring的xml檔案,<bean>中的name屬性是class屬性的一個別名,class屬性指類的全名,因為在SpringAction中有一個公共屬性Springdao,所以要在<bean>標籤中建立一個<property>標籤指定SpringDao。<property>標籤中的name就是SpringAction類中的SpringDao屬性名,ref指下面<bean name="springDao"...>,這樣其實是spring將SpringDaoImpl物件例項化並且呼叫SpringAction的setSpringDao方法將SpringDao注入:

xml程式碼  

<!--配置bean,配置後該類由spring管理-->  
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction">  
        <!--(1)依賴注入,配置當前類中相應的屬性-->  
        <property name="springDao" ref="springDao"></property>  
    </bean>  
<bean name="springDao" class="com.bless.springdemo.dao.impl.SpringDaoImpl"></bean>  

三:構造器注入

這種方式的注入是指帶有引數的建構函式注入,看下面的例子,我建立了兩個成員變數SpringDao和User,但是並未設定物件的set方法,所以就不能支援第一種注入方式,這裡的注入方式是在SpringAction的建構函式中注入,也就是說在建立SpringAction物件時要將SpringDao和User兩個引數值傳進來:

Java程式碼  

public class SpringAction {  
    //注入物件springDao  
    private SpringDao springDao;  
    private User user;  
      
    public SpringAction(SpringDao springDao,User user){  
        this.springDao = springDao;  
        this.user = user;  
        System.out.println("構造方法呼叫springDao和user");  
    }  
          
        public void save(){  
        user.setName("卡卡");  
        springDao.save(user);  
    }  
}  

在XML檔案中同樣不用<property>的形式,而是使用<constructor-arg>標籤,ref屬性同樣指向其它<bean>標籤的name屬性:

Xml程式碼  

<!--配置bean,配置後該類由spring管理-->  
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction">  
        <!--(2)建立構造器注入,如果主類有帶參的構造方法則需新增此配置-->  
        <constructor-arg ref="springDao"></constructor-arg>  
        <constructor-arg ref="user"></constructor-arg>  
    </bean>  
        <bean name="springDao" class="com.bless.springdemo.dao.impl.SpringDaoImpl"></bean>  
         <bean name="user" class="com.bless.springdemo.vo.User"></bean>  

  解決構造方法引數的不確定性,你可能會遇到構造方法傳入的兩引數都是同類型的,為了分清哪個該賦對應值,則需要進行一些小處理:

下面是設定index,就是引數位置:

Xml程式碼  :

<bean name="springAction" class="com.bless.springdemo.action.SpringAction">  
        <constructor-arg index="0" ref="springDao"></constructor-arg>  
        <constructor-arg index="1" ref="user"></constructor-arg>  
    </bean> 

另一種是設定引數型別:

Xml程式碼  :

<constructor-arg type="java.lang.String" ref=""/>  
  • 四丶靜態工廠的方法注入

靜態工廠顧名思義,就是通過呼叫靜態工廠的方法來獲取自己需要的物件,為了讓spring管理所有物件,我們不能直接通過"工程類.靜態方法()"來獲取物件,而是依然通過spring注入的形式獲取:

Java程式碼  

package com.bless.springdemo.factory;  
  
import com.bless.springdemo.dao.FactoryDao;  
import com.bless.springdemo.dao.impl.FactoryDaoImpl;  
import com.bless.springdemo.dao.impl.StaticFacotryDaoImpl;  
  
public class DaoFactory {  
    //靜態工廠  
    public static final FactoryDao getStaticFactoryDaoImpl(){  
        return new StaticFacotryDaoImpl();  
    }  
}  

同樣看關鍵類,這裡我需要注入一個FactoryDao物件,這裡看起來跟第一種注入一模一樣,但是看隨後的xml會發現有很大差別:

Java程式碼  :

 public class SpringAction {  
        //注入物件  
    private FactoryDao staticFactoryDao;  
      
    public void staticFactoryOk(){  
        staticFactoryDao.saveFactory();  
    }  
    //注入物件的set方法  
    public void setStaticFactoryDao(FactoryDao staticFactoryDao) {  
        this.staticFactoryDao = staticFactoryDao;  
    }  
}  

Spring的IOC配置檔案,注意看<bean name="staticFactoryDao">指向的class並不是FactoryDao的實現類,而是指向靜態工廠DaoFactory,並且配置 factory-method="getStaticFactoryDaoImpl"指定呼叫哪個工廠方法:

Xml程式碼  

<!--配置bean,配置後該類由spring管理-->  
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction" >  
        <!--(3)使用靜態工廠的方法注入物件,對應下面的配置檔案(3)-->  
        <property name="staticFactoryDao" ref="staticFactoryDao"></property>  
                </property>  
    </bean>  
    <!--(3)此處獲取物件的方式是從工廠類中獲取靜態方法-->  
    <bean name="staticFactoryDao" class="com.bless.springdemo.factory.DaoFactory" factory-method="getStaticFactoryDaoImpl"></bean>  

五丶例項工廠的方法注入

例項工廠的意思是獲取物件例項的方法不是靜態的,所以你需要首先new工廠類,再呼叫普通的例項方法:

Java程式碼  

public class DaoFactory {  
    //例項工廠  
    public FactoryDao getFactoryDaoImpl(){  
        return new FactoryDaoImpl();  
    }  
}  

那麼下面這個類沒什麼說的,跟前面也很相似,但是我們需要通過例項工廠類建立FactoryDao物件:

Java程式碼  

public class SpringAction {  
    //注入物件  
    private FactoryDao factoryDao;  
      
    public void factoryOk(){  
        factoryDao.saveFactory();  
    }  
  
    public void setFactoryDao(FactoryDao factoryDao) {  
        this.factoryDao = factoryDao;  
    }  
} 

最後看spring配置檔案:

Xml程式碼  

<!--配置bean,配置後該類由spring管理-->  
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction">  
        <!--(4)使用例項工廠的方法注入物件,對應下面的配置檔案(4)-->  
        <property name="factoryDao" ref="factoryDao"></property>  
    </bean>  
      
    <!--(4)此處獲取物件的方式是從工廠類中獲取例項方法-->  
    <bean name="daoFactory" class="com.bless.springdemo.factory.DaoFactory"></bean>  
    <bean name="factoryDao" factory-bean="daoFactory" factory-method="getFactoryDaoImpl"></bean> 

​​​​​​​平常的java開發中,程式設計師在某個類中需要依賴其它類的方法,則通常是new一個依賴類再呼叫類例項的方法,這種開發存在的問題是new的類例項不好統一管理,spring提出了依賴注入的思想,即依賴類不由程式設計師例項化,而是通過spring容器幫我們new指定例項並且將例項注入到需要該物件的類中。依賴注入的另一種說法是“控制反轉”,通俗的理解是:平常我們new一個例項,這個例項的控制權是我們程式設計師,而控制反轉是指new例項工作不由我們程式設計師來做而是交給spring容器來做。