spring不但支援自己定義的@Autowired注解,還支援幾個由JSR-250規範定義的注解,它們分别是@Resource、
@PostConstruct以及@PreDestroy。
一[email protected]
@Resource的作用相當于@Autowired,隻不過@Autowired按byType自動注入,而@Resource預設按 byName自動注入
罷了。@Resource有兩個屬性是比較重要的,分是name和type,Spring将@Resource注解的name屬性解析為bean的名字
,而type屬性則解析為bean的類型。是以如果使用name屬性,則使用byName的自動注入政策,而使用type屬性時則使
用byType自動注入政策。如果既不指定name也不指定type屬性,這時将通過反射機制使用byName自動注入政策。
@Resource裝配順序
1. 如果同時指定了name和type,則從Spring上下文中找到唯一比對的bean進行裝配,找不到則抛出異常
2. 如果指定了name,則從上下文中查找名稱(id)比對的bean進行裝配,找不到則抛出異常
3. 如果指定了type,則從上下文中找到類型比對的唯一bean進行裝配,找不到或者找到多個,都會抛出異常
4. 如果既沒有指定name,又沒有指定type,則自動按照byName方式進行裝配;如果沒有比對,則回退為一個原始
類型進行比對,如果比對則自動裝配;
二[email protected] 與@Resource的差別:
1、 @Autowired與@Resource都可以用來裝配bean. 都可以寫在字段上,或寫在setter方法上。
2、 @Autowired預設按類型裝配(這個注解是屬業spring的),預設情況下必須要求依賴對象必須存在,如果要允許
null值,可以設定它的required屬性為false,如:@Autowired(required=false) ,如果我們想使用名稱裝配可以結
合@Qualifier注解進行使用,如下:
@Autowired
@Qualifier("baseDao")
private BaseDao baseDao;
如果直接使用
@Autowired
private BaseDao baseDao;
那麼其預設的beanName為baseDaoImpl(實作類名字,開頭小寫);
@Resource 預設的name也是一樣的;
其中IDEA可以識别标注的name存不存在;(編譯檢查)
是以在接口隻要一個實作類的情況下,可以直接使用@Autowired
3、@Resource(這個注解屬于J2EE的),預設按照名稱進行裝配,名稱可以通過name屬性進行指定,如果沒有指定
name屬性,當注解寫在字段上時,預設取字段名進行安裝名稱查找,如果注解寫在setter方法上預設取屬性名進行裝
配。當找不到與名稱比對的bean時才按照類型進行裝配。但是需要注意的是,如果name屬性一旦指定,就隻會按照名
稱進行裝配。
注:預設情況可能會2次查找,性能會低些;
三 @qualifier作用
1.限定比對bean
2.也支援邏輯限定,即可以修飾注解,如@LoadBalanced
詳見:https://www.cnblogs.com/fangshixiang/p/11532717.html
四[email protected] :優先考慮注入的類,比如兩個類繼承了同一個接口,然後注入的時候是以接口名的形式注入到另一個
類中的。 這個時候會報一下錯誤:
Unsatisfied dependency expressed through field ‘animal’; nested exception is
org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type
‘com.hbr.three.Animal’ available: expected single matching bean but found 2: dog,panda
也就是說,animal接口沒有唯一的bean定義。本來想找一個bean的,可是卻發現了兩個,dog和panda. 然後就不知道
帶注入哪個執行個體了,然後就報錯了。
被注入的類是這樣的:
@Component
public class DIclass {
@Autowired
private Animal animal;
public void eat(){
animal.eat();
}
}
然後dog和panda實作了animal的接口。
測試裡面是這樣的:
AnnotationConfigApplicationContext app = new AnnotationConfigApplicationContext(Configurate.class);
DIclass bean = (DIclass)app.getBean(“DIclass”);
bean.eat();
這裡注意,包掃描的beanname是類名一緻,沒有首字母小寫的,不要和@bean注解預設的名字搞混了。
是以這個時候 @Primary 注解就發揮作用了,它會優先考慮有這個注解的類。
參考:https://www.cnblogs.com/fengli9998/p/7472247.html