6、Bean的自動裝配
6.1 自動裝配說明
-
自動裝配是使用spring滿足bean依賴的一種方法
-
spring會在應用上下文中為某個bean尋找其依賴的bean。
Spring中bean的三種裝配機制:
-
在xml中顯式配置;
-
在java中顯式配置;
-
隱式的bean發現機制和自動裝配。
這裡主要看:自動化的裝配bean。
Spring的自動裝配需要從兩個角度來實現,或者說是兩個操作:
-
元件掃描(component scanning):spring會自動發現應用上下文中所建立的bean;
-
自動裝配(autowiring):spring自動滿足bean之間的依賴,也就是我們說的IoC/DI;
元件掃描和自動裝配組合發揮巨大威力,使得顯示的配置降低到最少。
6.2 測試環境搭建
1、新建一個專案
2、新建兩個實體類,Cat Dog 都有一個叫的方法
public class Cat { public void shout() { System.out.println("miao~"); } } public class Dog { public void shout() { System.out.println("wang~"); } }
3、新建一個使用者類 User
public class User { private Cat cat; private Dog dog; private String str; }
4、編寫Spring配置檔案
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <bean id="dog" class="com.chen.pojo.Dog"/> <bean id="cat" class="com.chen.pojo.Cat"/> <bean id="user" class="com.chen.pojo.User"> <property name="cat" ref="cat"/> <property name="dog" ref="dog"/> <property name="str" value="chen"/> </bean> </beans>
5、測試
public class MyTest { @Test public void testMethodAutowire() { ApplicationContext context = newClassPathXmlApplicationContext("beans.xml"); User user = (User) context.getBean("user"); user.getCat().shout(); user.getDog().shout(); } }
6.3 byName自動裝配
autowire byName (按名稱自動裝配)
由於在手動配置xml過程中,常常發生字母缺漏和大小寫等錯誤,而無法對其進行檢查,使得開發效率降低。
採用自動裝配將避免這些錯誤,並且使配置簡單化。
測試:
1、修改bean配置,增加一個屬性 autowire="byName"
<bean id="user" class="com.chen.pojo.User" autowire="byName"> <property name="str" value="chen"/> </bean>
2、再次測試,結果依舊成功輸出!
3、我們將 cat 的bean id修改為 cat11
4、再次測試, 執行時報空指標 java.lang.NullPointerException 。因為按byName規則找不對應set方法,真正的setCat就沒執行,物件就沒有初始化,所以呼叫時就會報空指標錯誤。
小結:
當一個bean節點帶有 autowire byName的屬性時。
-
將查詢其類中所有的set方法名,例如setCat,獲得將set去掉並且首字母小寫的字串,即cat。
-
去spring容器中尋找是否有此字串名稱id的物件。
-
如果有,就取出注入;如果沒有,就報空指標異常。
6.4 byType自動裝配
autowire byType (按型別自動裝配)
使用autowire byType首先需要保證:同一型別的物件,在spring容器中唯一。如果不唯一,會報不唯一的異常 NoUniqueBeanDefinitionException 。
測試:
1、將user的bean配置修改一下 : autowire="byType"
2、測試,正常輸出
3、在註冊一個cat 的bean物件!
<bean id="dog" class="com.chen.pojo.Dog"/> <bean id="cat" class="com.chen.pojo.Cat"/> <bean id="cat2" class="com.chen.pojo.Cat"/> <bean id="user" class="com.chen.pojo.User" autowire="byType"> <property name="str" value="chen"/> </bean>
4、測試,報錯:NoUniqueBeanDefinitionException
5、刪掉cat2,將cat的bean名稱改掉!測試!因為是按型別裝配,所以並不會報異常,也不影響最後的結果。甚至將id屬性去掉,也不影響結果。
這就是按照型別自動裝配。
6.5 使用註解實現自動裝配
使用註解
jdk1.5開始支援註解,spring2.5開始全面支援註解。
準備工作:利用註解的方式注入屬性。
<?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" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"> <!--開啟註解支援--> <context:annotation-config/> </beans>
@Autowired
-
@Autowired是按型別自動裝配的,不支援id匹配。
-
需要匯入 spring-aop的包!
測試:
1、將User類中的set方法去掉,使用@Autowired註解
public class User { @Autowired private Cat cat; @Autowired private Dog dog; private String str; public Cat getCat() { return cat;
} public Dog getDog() { return dog; } public String getStr() { return str; } }
2、此時配置檔案內容
<?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" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
<context:annotation-config/> <bean id="dog" class="com.chen.pojo.Dog"/> <bean id="cat" class="com.chen.pojo.Cat"/> <bean id="user" class="com.chen.pojo.User"/>
</beans>
3、測試,成功輸出結果。
@Autowired(required=false)
說明:false,物件可以為null;true,物件必須存物件,不能為null。
//如果允許物件為null,設定required = false,預設為true @Autowired(required = false) private Cat cat;
@Qualifier
-
@Autowired是根據型別自動裝配的,加上@Qualifier則可以根據byName的方式自動裝配
-
@Qualifier不能單獨使用。
測試
1、配置檔案修改內容,保證型別存在物件。且名字不為類的預設名字!
<bean id="dog1" class="com.chen.pojo.Dog"/> <bean id="dog2" class="com.chen.pojo.Dog"/> <bean id="cat1" class="com.chen.pojo.Cat"/> <bean id="cat2" class="com.chen.pojo.Cat"/>
2、沒有加Qualifier測試,直接報錯
3、在屬性上新增Qualifier註解
@Autowired @Qualifier(value = "cat2") private Cat cat; @Autowired @Qualifier(value = "dog2") private Dog dog;
測試,成功輸出。
@Resource
-
@Resource如有指定的name屬性,先按該屬性進行byName方式查詢裝配;
-
其次再進行預設的byName方式進行裝配;
-
如果以上都不成功,則按byType的方式自動裝配。
-
都不成功,則報異常。
實體類:
public class User { @Resource(name = "cat2") private Cat cat; @Resource private Dog dog; private String str; }
xml:
<bean id="dog" class="com.chen.pojo.Dog"/> <bean id="cat1" class="com.chen.pojo.Cat"/> <bean id="cat2" class="com.chen.pojo.Cat"/> <bean id="user" class="com.chen.pojo.User"/>
測試成功。
刪掉cat2
<bean id="dog" class="com.chen.pojo.Dog"/> <bean id="cat1" class="com.chen.pojo.Cat"/> <bean id="user" class="com.chen.pojo.User"/>
實體類上只保留註解
@Resource private Cat cat; @Resource private Dog dog;
測試成功。
小結
@Autowired與@Resource異同:
1、@Autowired與@Resource都可以用來裝配bean。都可以寫在欄位上,或寫在setter方法上。
2、@Autowired預設按型別裝配(屬於spring規範),預設情況下必須要求依賴物件必須存在,如果要允許null 值,可以設定它的required屬性為false,如:@Autowired(required=false) ,如果想使用名稱裝配可以結合@Qualifier註解進行使用
3、@Resource(屬於J2EE復返),預設按照名稱進行裝配,名稱可以通過name屬性進行指定。如果沒有指定name屬性,當註解寫在欄位上時,預設取欄位名進行按照名稱查詢,如果註解寫在setter方法上預設取屬性名進行裝配。當找不到與名稱匹配的bean時才按照型別進行裝配。但是需要注意的是,如果name屬性一旦指定,就只會按照名稱進行裝配。
它們的作用相同都是用註解方式注入物件,但執行順序不同。@Autowired先byType,@Resource先byName。