spring的五種依賴注入方式
平常的java開發中,程式設計師在某個類中需要依賴其它類的方法,則通常是new一個依賴類再呼叫類例項的方法,這種開發存在的問題是new的類例項不好統一管理,spring提出了依賴注入的思想,即依賴類不由程式設計師例項化,而是通過spring容器幫我們new指定例項並且將例項注入到需要該物件的類中。依賴注入的另一種說法是“控制反轉”,通俗的理解是:平常我們new一個例項,這個例項的控制權是我們程式設計師,而控制反轉是指new例項工作不由我們程式設計師來做而是交給spring容器來做。
spring有多種依賴注入的形式,下面介紹spring進行DI的方式:
一.目前使用最廣泛的 @Autowired:自動裝配
自動裝配,用於替代基於XML配置的自動裝配,詳見【3.3.3 自動裝配】。
基於@Autowired的自動裝配,預設是根據型別注入,可以用於構造器、欄位、方法注入,使用方式如下:
java程式碼:
- @Autowired(required=true)
- 構造器、欄位、方法
@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容器來做。