面試必問(wèn):@Autowired 和 @Resource 有什么區(qū)別?
@Autowired 和 @Resource 都是 Spring/Spring Boot 項(xiàng)目中,用來(lái)進(jìn)行依賴(lài)注入的注解。它們都提供了將依賴(lài)對(duì)象注入到當(dāng)前對(duì)象的功能,但二者卻有眾多不同,并且這也是常見(jiàn)的面試題之一,所以我們今天就來(lái)盤(pán)它。@Autowired 和 @Resource 的區(qū)別主要體現(xiàn)在以下 5 點(diǎn):
- 來(lái)源不同;
- 依賴(lài)查找的順序不同;
- 支持的參數(shù)不同;
- 依賴(lài)注入的支持不同;
- 編譯器 IDEA 的提示不同。
1、來(lái)源不同
@Autowired 和 @Resource 來(lái)自不同的“父類(lèi)”,其中 @Autowired 是 Spring 定義的注解,而 @Resource 是 Java 定義的注解,它來(lái)自于 JSR-250(Java 250 規(guī)范提案)。
小知識(shí):JSR 是 Java Specification Requests 的縮寫(xiě),意思是“Java 規(guī)范提案”。任何人都可以提交 JSR 給 Java 官方,但只有最終確定的 JSR,才會(huì)以 JSR-XXX 的格式發(fā)布,如 JSR-250,而被發(fā)布的 JSR 就可以看作是 Java 語(yǔ)言的規(guī)范或標(biāo)準(zhǔn)。
2、依賴(lài)查找順序不同
依賴(lài)注入的功能,是通過(guò)先在 Spring IoC 容器中查找對(duì)象,再將對(duì)象注入引入到當(dāng)前類(lèi)中。而查找有分為兩種實(shí)現(xiàn):按名稱(chēng)(byName)查找或按類(lèi)型(byType)查找,其中 @Autowired 和 @Resource 都是既使用了名稱(chēng)查找又使用了類(lèi)型查找,但二者進(jìn)行查找的順序卻截然相反。
(1)@Autowired 查找順序
@Autowired 是先根據(jù)類(lèi)型(byType)查找,如果存在多個(gè) Bean 再根據(jù)名稱(chēng)(byName)進(jìn)行查找,它的具體查找流程如下:
關(guān)于以上流程,可以通過(guò)查看 Spring 源碼中的 org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor#postProcessPropertyValues 實(shí)現(xiàn)分析得出,源碼執(zhí)行流程如下圖所示:
(2)@Resource 查找順序
@Resource 是先根據(jù)名稱(chēng)查找,如果(根據(jù)名稱(chēng))查找不到,再根據(jù)類(lèi)型進(jìn)行查找,它的具體流程如下圖所示:
關(guān)于以上流程可以在 Spring 源碼的 org.springframework.context.annotation.CommonAnnotationBeanPostProcessor#postProcessPropertyValues 中分析得出。雖然 @Resource 是 JSR-250 定義的,但是由 Spring 提供了具體實(shí)現(xiàn),它的源碼實(shí)現(xiàn)如下:
(3)查找順序小結(jié)
由上面的分析可以得出:
- @Autowired 先根據(jù)類(lèi)型(byType)查找,如果存在多個(gè)(Bean)再根據(jù)名稱(chēng)(byName)進(jìn)行查找;
- @Resource 先根據(jù)名稱(chēng)(byName)查找,如果(根據(jù)名稱(chēng))查找不到,再根據(jù)類(lèi)型(byType)進(jìn)行查找。
3、支持的參數(shù)不同
@Autowired 和 @Resource 在使用時(shí)都可以設(shè)置參數(shù),比如給 @Resource 注解設(shè)置 name 和 type 參數(shù),實(shí)現(xiàn)代碼如下:
但二者支持的參數(shù)以及參數(shù)的個(gè)數(shù)完全不同,其中 @Autowired 只支持設(shè)置一個(gè) required 的參數(shù),而 @Resource 支持 7 個(gè)參數(shù),支持的參數(shù)如下圖所示:
4、依賴(lài)注入的支持不同
@Autowired 和 @Resource 支持依賴(lài)注入的用法不同,常見(jiàn)依賴(lài)注入有以下 3 種實(shí)現(xiàn):
- 屬性注入
- 構(gòu)造方法注入
- Setter 注入
這 3 種實(shí)現(xiàn)注入的實(shí)現(xiàn)代碼如下。
a) 屬性注入
b) 構(gòu)造方法注入
c) Setter 注入
其中,@Autowired 支持屬性注入、構(gòu)造方法注入和 Setter 注入,而 @Resource 只支持屬性注入和 Setter 注入,當(dāng)使用 @Resource 實(shí)現(xiàn)構(gòu)造方法注入時(shí)就會(huì)提示以下錯(cuò)誤:
5、編譯器提示不同
當(dāng)使用 IDEA 專(zhuān)業(yè)版在編寫(xiě)依賴(lài)注入的代碼時(shí),如果注入的是 Mapper 對(duì)象,那么使用 @Autowired 編譯器會(huì)提示報(bào)錯(cuò)信息,報(bào)錯(cuò)內(nèi)容如下圖所示:
雖然 IDEA 會(huì)出現(xiàn)報(bào)錯(cuò)信息,但程序是可以正常執(zhí)行的。然后,我們?cè)賹⒁蕾?lài)注入的注解更改為 @Resource 就不會(huì)出現(xiàn)報(bào)錯(cuò)信息了,具體實(shí)現(xiàn)如下:
總結(jié)
@Autowired 和 @Resource 都是用來(lái)實(shí)現(xiàn)依賴(lài)注入的注解(在 Spring/Spring Boot 項(xiàng)目中),但二者卻有著 5 點(diǎn)不同:
- 來(lái)源不同:@Autowired 來(lái)自 Spring 框架,而 @Resource 來(lái)自于(Java)JSR-250;
- 依賴(lài)查找的順序不同:@Autowired 先根據(jù)類(lèi)型再根據(jù)名稱(chēng)查詢(xún),而 @Resource 先根據(jù)名稱(chēng)再根據(jù)類(lèi)型查詢(xún);
- 支持的參數(shù)不同:@Autowired 只支持設(shè)置 1 個(gè)參數(shù),而 @Resource 支持設(shè)置 7 個(gè)參數(shù);
- 依賴(lài)注入的用法支持不同:@Autowired 既支持構(gòu)造方法注入,又支持屬性注入和 Setter 注入,而 @Resource 只支持屬性注入和 Setter 注入;
- 編譯器 IDEA 的提示不同:當(dāng)注入 Mapper 對(duì)象時(shí),使用 @Autowired 注解編譯器會(huì)提示錯(cuò)誤,而使用 @Resource 注解則不會(huì)提示錯(cuò)誤。
參考 & 鳴謝
- www.cnblogs.com/felordcn/p/13063802.html
- blog.csdn.net/CPLASF_/article/details/109225213