@Autowired 與@Resource選擇(治好你的強迫症)


spring不但支持自己定義的@Autowired注解,還支持幾個由JSR-250規范定義的注解,它們分別是@Resource、@PostConstruct以及@PreDestroy。
  @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方式進行裝配;如果沒有匹配,則回退為一個原始類型進行匹配,如果匹配則自動裝配;

 

@Autowired 與@Resource的區別:

1、 @Autowired與@Resource都可以用來裝配bean. 都可以寫在字段上,或寫在setter方法上。

2、 @Autowired默認按類型裝配(這個注解是屬業spring的),默認情況下必須要求依賴對象必須存在,如果要允許null值,可以設置它的required屬性為false,如:@Autowired(required=false) ,如果我們想使用名稱裝配可以結合@Qualifier注解進行使用,如下:

@Autowired()
@Qualifier("baseDao")
privateBaseDao baseDao;

3、@Resource(這個注解屬於J2EE的),默認按照名稱進行裝配,名稱可以通過name屬性進行指定,如果沒有指定name屬性,當注解寫在字段上時,默認取字段名進行安裝名稱查找,如果注解寫在setter方法上默認取屬性名進行裝配。當找不到與名稱匹配的bean時才按照類型進行裝配。但是需要注意的是,如果name屬性一旦指定,就只會按照名稱進行裝配。

推薦使用:@Resource注解在字段上,這樣就不用寫setter方法了,並且這個注解是屬於J2EE的,減少了與spring的耦合。這樣代碼看起就比較優雅。

 

 

 

 

@Qualifier注解的作用

@Autowired是根據類型進行自動裝配的。如果當Spring上下文中存在不止一個UserDao類型的bean時,就會拋出BeanCreationException異常;如果Spring上下文中不存在UserDao類型的bean,也會拋出BeanCreationException異常。我們可以使用@Qualifier配合@Autowired來解決這些問題。如下:

①可能存在多個UserDao實例

@Autowired 
@Qualifier("userServiceImpl") 
public IUserService userService; 

或者

@Autowired
public void setUserDao(@Qualifier("userDao") UserDao userDao) {
  this.userDao = userDao;
}

這樣Spring會找到id為userServiceImpl和userDao的bean進行裝配。

②可能不存在UserDao實例

@Autowired(required = false) 
public IUserService userService

個人總結:

@Autowired 根據類型type注入, 

@Qualifier("cusInfoService")//一般作為@Autowired()的修飾用,

@Autowired @Qualifie("userService") 兩個結合起來可以根據名字和類型注入

@Resource 默認根據名字name注入,其次按照類型搜索,也可以通過name和type屬性進行選擇性注入

一般@Autowired和@Qualifier一起用,@Resource單獨用。

當然沒有沖突的話@Autowired也可以單獨用

 

復雜理解:

比如你有這么一個Bean

@Service(“UserService”)

public Class UserServiceImpl implements UserService{};

現在你想在UserController 里面使用這個UserServiceImpl 

public Class UserController {}

 

@AutoWire  

當使用這個注入的時候上面的 UserServiceImpl 只需要這樣寫 @Service,這樣就會自動找到UserService這個類型以及他的子類型。UserServiceImpl 實現了UserService,所以能夠找到它。

缺點:就是當UserService實現類有兩個以上的時候,這就造成了沖突,所以要用@AutoWire注入的時候要確保UserService只有一個實現類。

 

@Resource

默認情況下是按照名稱進行匹配,如果沒有找到相同名稱的Bean,則會按照類型進行匹配,有人可能會想了,這下好了,用這個是萬能的了,不用管名字了,也不用管類型了,

缺點:

  根據這個注解的匹配效果可以看出,它進行了兩次匹配,也就是說,如果你在UserService這個類上面這樣寫注解,@Service,它會怎么找呢,首先是找相同名字的,如果沒有找到,再找相同類型的,而這里的@Service沒有寫名字,這個時候就進行了兩次搜索,顯然,速度就下降了許多。也許你還會問,這里的@Service本來就沒有名字,肯定是直接進行類型搜索啊。其實不是這樣的,UserServiceImpl 上面如果有@Service默認的名字 是這個userServiceImpl,注意看,就是把類名前面的大寫變成小寫,就是默認的Bean的名字了。一般開發都會自己指定名字,例如:userService

  @Resource根據名字搜索是這樣寫@Resource("userService"),如果你寫了這個名字叫userService,那么UserServiceImpl上面必須也是這個名字,不然還是會報錯。

 

@Autowired @Qualifie("userService") 

是直接按照名字進行搜索,也就是說,對於UserServiceImpl 上面@Service注解必須寫名字,不寫就會報錯,而且名字必須是@Autowired @Qualifie("userService") 保持一致。如果@Service上面寫了名字,而@Autowired @Qualifie() ,一樣會報錯。

 

說了這么多,可能你有些說暈了,那么怎么用這三個呢,

要實際的工作是根據實際情況來使用的,UserServiceImpl上面可能會這樣寫 @Service("userService"),這樣通常使用AutoWire和@Resource多一些,bean的名字不用寫。

這里的實際工作情況,到底是什么情況呢?

如果你的架構設計師考慮的比較精細,要求比較嚴格,要求項目上線后的訪問速度比較好,這個時候@AutoWire沒有@Resource好用,因為@Resource可以根據名字來搜索,是這樣寫的@Resource(name="userService"),可以直接寫@Resource("userService"),因為默認byname。

為什么推薦@Resource(name="userService")???

因為根據名字搜索是最快的,就好像查數據庫一樣,根據Id查找最快。因為這里的名字與數據庫里面的ID是一樣的作用。這個時候,就要求你多寫幾個名字,工作量自然就增加了。而如果你不用注解,用xml文件的時候,對於注入Bean的時候要求寫一個Id,xml文件時候的id就相當於這里的名字。

 

這個@Autowired @Qualifie("userService") 也可以用名字啊,為什么不用呢?

原因很簡單,這個有點長,不喜歡,增加工作量。

 

說了那么多沒用,你能做的就是簡單直接,什么最方便就用什么,

你就直接用@Resource得了,如果你喜歡用@Autowire也行,不用寫名字。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM