天天看點

spring @Resource @Autowired @qualifier

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

繼續閱讀