1. 程式人生 > >@Autowired 與@Resource的區別(詳細)

@Autowired 與@Resource的區別(詳細)

不用 數據庫 聲明式 構造 eve ring 自動註入 post 方法

參考博文:

http://www.cnblogs.com/happyyang/articles/3553687.html

http://blog.csdn.net/revent/article/details/49203619

http://blog.csdn.net/ad921012/article/details/49679745

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註解進行使用,如下:

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

1
2
@Resource(name="baseDao")
privateBaseDao baseDao;
推薦使用:@Resource註解在字段上,這樣就不用寫setter方法了,並且這個註解是屬於J2EE的,減少了與spring的耦合。這樣代碼看起就比較優雅。

spring @Qualifier註解

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

①可能存在多個UserDao實例

[java] view plaincopy在CODE上查看代碼片派生到我的代碼片

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

或者

[java] view plaincopy在CODE上查看代碼片派生到我的代碼片

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

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

②可能不存在UserDao實例

[java] view plaincopy在CODE上查看代碼片派生到我的代碼片

@Autowired(required = false)
public IUserService userService

個人總結:

@Autowired//默認按type註入
@Qualifier("cusInfoService")//一般作為@Autowired()的修飾用
@Resource(name="cusInfoService")//默認按name註入,可以通過name和type屬性進行選擇性註入

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

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

-----------常用註解--------

--定義Bean的註解

@Controller

@Controller("Bean的名稱")

定義控制層Bean,如Action

@Service

@Service("Bean的名稱")

定義業務層Bean

@Repository

@Repository("Bean的名稱")

定義DAO層Bean

@Component

定義Bean, 不好歸類時使用.

--自動裝配Bean (選用一種註解就可以)

@Autowired (Srping提供的)

默認按類型匹配,自動裝配(Srping提供的),可以寫在成員屬性上,或寫在setter方法上

@Autowired(required=true)

一定要找到匹配的Bean,否則拋異常。 默認值就是true

@Autowired

@Qualifier("bean的名字")

按名稱裝配Bean,與@Autowired組合使用,解決按類型匹配找到多個Bean問題。

@Resource JSR-250提供的

默認按名稱裝配,當找不到名稱匹配的bean再按類型裝配.

可以寫在成員屬性上,或寫在setter方法上

可以通過@Resource(name="beanName") 指定被註入的bean的名稱, 要是未指定name屬性, 默認使用成員屬性的變量名,一般不用寫name屬性.

@Resource(name="beanName")指定了name屬性,按名稱註入但沒找到bean, 就不會再按類型裝配了.

@Inject 是JSR-330提供的

按類型裝配,功能比@Autowired少,沒有使用的必要。

--定義Bean的作用域和生命過程

@Scope("prototype")

值有:singleton,prototype,session,request,session,globalSession

@PostConstruct

相當於init-method,使用在方法上,當Bean初始化時執行。

@PreDestroy

相當於destory-method,使用在方法上,當Bean銷毀時執行。

--聲明式事務

@Transactional

@Autowired @Resource @Qualifier的區別

實用理解:@Autowired @Resource 二選其一,看中哪個就用哪個。

簡單理解:

@Autowired 根據類型註入,

@Resource 默認根據名字註入,其次按照類型搜索

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

復雜理解:

比如你有這麽一個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的名字了。 @Resource根據名字搜索是這樣寫@Resource("userService"),如果你寫了這個名字叫userService,那麽UserServiceImpl上面必須也是這個名字,不然還是會報錯。

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

private UserService userService;

說了這麽多,可能你有些說暈了,那麽怎麽用這三個呢,要實際的工作是根據實際情況來使用的,通常使用AutoWire和@Resource多一些,bean的名字不用寫,而UserServiceImpl上面能會這樣寫 @Service("userService")。這裏的實際工作情況,到底是什麽情況呢?說白了就是整個項目設計時候考慮的情況,如果你的架構設計師考慮的比較精細,要求比較嚴格,要求項目上線後的訪問速度比較好,通常是考慮速度了。這個時候@AutoWire沒有@Resource好用,因為@Resource可以根據名字來搜索,是這樣寫的@Resource("userService")。這個@Autowired @Qualifie("userService") 也可以用名字啊,為什麽不用呢,原因很簡單,這個有點長,不喜歡,增加工作量。因為根據名字搜索是最快的,就好像查數據庫一樣,根據Id查找最快。因為這裏的名字與數據庫裏面的ID是一樣的作用。這個時候,就要求你多寫幾個名字,工作量自然就增加了。而如果你不用註解,用xml文件的時候,對於註入Bean的時候要求寫一個Id,xml文件時候的id就相當於這裏的名字。

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

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

通常情況一個Bean的註解寫錯了,會報下面這些錯誤,最為常見,

No bean named ‘user‘ is defined,這個表示沒有找到被命名為user的Bean,通俗的說,就是名字為user的類型,以及它的子類型,出現這個錯誤的原因就是註入時候的類型名字為user,而搜索的時候找不到,也就是說可能那個搜索的類型,並沒有命令為user,解決辦法就是找到這個類型,去命令為user,

下面這個錯誤也常見,

No qualifying bean of type [com.service.UserService] found for dependency:

這個錯誤的原因就是類型上面沒有加@Service這個註入,不僅僅是@Service,如果是其他層也會出現這個錯誤,這裏我是以Service為例子說明,如果是DAO層就是沒有加@Repository,Controller層,則是沒有加@Controller。

還有,如果你還是想再簡單點,無論是DAO,Controller,Service三個層,都可以用這個註解,@Component,這個註解通用所有的Bean,這個時候你可能會說了,有通常的為什麽用的人少呢,那是因為MVC這個分層的設計原則,用@Repository,@Service,@Controller,這個可以區別MVC原則中的DAO,Service,Controller。便於識別。

博客2:

springautowiredqualifierbytypebyname
 在使用Spring框架中@Autowired標簽時默認情況下使用
Java代碼
@Autowired
@Autowired
註釋進行自動註入時,Spring 容器中匹配的候選 Bean 數目必須有且僅有一個。當找不到一個匹配的 Bean 時,Spring 容器將拋出 BeanCreationException 異常,並指出必須至少擁有一個匹配的 Bean。
@Autowired 默認是按照byType進行註入的,如果發現找到多個bean,則,又按照byName方式比對,如果還有多個,則報出異常。

例子:

@Autowired
private ExamUserMapper examUserMapper; - ExamUserMapper是一個接口

1. spring先找類型為ExamUserMapper的bean

2. 如果存在且唯一,則OK;

3. 如果不唯一,在結果集裏,尋找name為examUserMapper的bean。因為bean的name有唯一性,所以,到這裏應該能確定是否存在滿足要求的bean了

@Autowired也可以手動指定按照byName方式註入,使用@Qualifier標簽,例如:
@Autowired () @Qualifier ( "baseDao" )


  Spring 允許我們通過
Java代碼
@Qualifier
@Qualifier
註釋指定註入 Bean 的名稱,這樣歧義就消除了,可以通過下面的方法解決異常。

  
Java代碼
@Qualifier("XXX")
@Qualifier("XXX")
中的 XX是 Bean 的名稱,所以 @Autowired 和 @Qualifier 結合使用時,自動註入的策略就從 byType 轉變成 byName 了。

  @Autowired 可以對成員變量、方法以及構造函數進行註釋,而 @Qualifier 的標註對象是成員變量、方法入參、構造函數入參。

Spring不但支持自己定義的@Autowired註解,還支持幾個由JSR-250規範定義的註解,它們分別是@Resource、@PostConstruct以及@PreDestroy。

  
Java代碼
@Resource
@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的區別(詳細)