首頁 > 軟體

一文搞懂Spring中@Autowired和@Resource的區別

2022-08-29 14:01:49

@Autowired 和 @Resource 都是 Spring/Spring Boot 專案中,用來進行依賴注入的註解。它們都提供了將依賴物件注入到當前物件的功能,但二者卻有眾多不同,並且這也是常見的面試題之一,所以我們今天就來盤它。

@Autowired 和 @Resource 的區別主要體現在以下 5 點:

  • 來源不同;
  • 依賴查詢的順序不同;
  • 支援的引數不同;
  • 依賴注入的用法不同;
  • 編譯器 IDEA 的提示不同。

1.來源不同

@Autowired 和 @Resource 來自不同的“父類別”,其中 @Autowired 是 Spring 定義的註解,而 @Resource 是 Java 定義的註解,它來自於 JSR-250(Java 250 規範提案)。

小知識:JSR 是 Java Specification Requests 的縮寫,意思是“Java 規範提案”。任何人都可以提交 JSR 給 Java 官方,但只有最終確定的 JSR,才會以 JSR-XXX 的格式釋出,如 JSR-250,而被髮布的 JSR 就可以看作是 Java 語言的規範或標準。

2.依賴查詢順序不同

依賴注入的功能,是通過先在 Spring IoC 容器中查詢物件,再將物件注入引入到當前類中。而查詢有分為兩種實現:按名稱(byName)查詢或按型別(byType)查詢,其中 @Autowired 和 @Resource 都是既使用了名稱查詢又使用了型別查詢,但二者進行查詢的順序卻截然相反。

2.1 @Autowired 查詢順序

@Autowired 是先根據型別(byType)查詢,如果存在多個 Bean 再根據名稱(byName)進行查詢,它的具體查詢流程如下:

關於以上流程,可以通過檢視 Spring 原始碼中的

org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor#postProcessPropertyValues 實現分析得出,原始碼執行流程如下圖所示:

2.2 @Resource 查詢順序

@Resource 是先根據名稱查詢,如果(根據名稱)查詢不到,再根據型別進行查詢,它的具體流程如下圖所示:

關於以上流程可以在 Spring 原始碼的 org.springframework.context.annotation.CommonAnnotationBeanPostProcessor#postProcessPropertyValues 中分析得出。雖然 @Resource 是 JSR-250 定義的,但是由 Spring 提供了具體實現,它的原始碼實現如下:

2.3 查詢順序小結

由上面的分析可以得出:

  • @Autowired 先根據型別(byType)查詢,如果存在多個(Bean)再根據名稱(byName)進行查詢;
  • @Resource 先根據名稱(byName)查詢,如果(根據名稱)查詢不到,再根據型別(byType)進行查詢。

3.支援的引數不同

@Autowired 和 @Resource 在使用時都可以設定引數,比如給 @Resource 註解設定 name 和 type 引數,實現程式碼如下:

@Resource(name = "userinfo", type = UserInfo.class)
private UserInfo user;

二者支援的引數以及引數的個數完全不同,其中 @Autowired 只支援設定一個 required 的引數,而 @Resource 支援 7 個引數,支援的引數如下圖所示:

4.依賴注入的支援不同

@Autowired 和 @Resource 支援依賴注入的用法不同,常見依賴注入有以下 3 種實現:

  • 屬性注入
  • 構造方法注入
  • Setter 注入

這 3 種實現注入的實現程式碼如下。

a) 屬性注入

@RestController
public class UserController {
    // 屬性注入
    @Autowired
    private UserService userService;

    @RequestMapping("/add")
    public UserInfo add(String username, String password) {
        return userService.add(username, password);
    }
}

b) 構造方法注入

@RestController
public class UserController {
    // 構造方法注入
    private UserService userService;

    @Autowired
    public UserController(UserService userService) {
        this.userService = userService;
    }

    @RequestMapping("/add")
    public UserInfo add(String username, String password) {
        return userService.add(username, password);
    }
}

c) Setter 注入

@RestController
public class UserController {
    // Setter 注入
    private UserService userService;

    @Autowired
    public void setUserService(UserService userService) {
        this.userService = userService;
    }

    @RequestMapping("/add")
    public UserInfo add(String username, String password) {
        return userService.add(username, password);
    }
}

其中, @Autowired 支援屬性注入、構造方法注入和 Setter 注入,而 @Resource 只支援屬性注入和 Setter 注入,當使用 @Resource 實現構造方法注入時就會提示以下錯誤:

5.編譯器提示不同

當使用 IDEA 專業版在編寫依賴注入的程式碼時,如果注入的是 Mapper 物件,那麼使用 @Autowired 編譯器會提示報錯資訊,報錯內容如下圖所示:

雖然 IDEA 會出現報錯資訊,但程式是可以正常執行的。

然後,我們再將依賴注入的註解更改為 @Resource 就不會出現報錯資訊了,具體實現如下:

總結

@Autowired 和 @Resource 都是用來實現依賴注入的註解(在 Spring/Spring Boot 專案中),但二者卻有著 5 點不同:

  • 來源不同:@Autowired 來自 Spring 框架,而 @Resource 來自於(Java)JSR-250;
  • 依賴查詢的順序不同:@Autowired 先根據型別再根據名稱查詢,而 @Resource 先根據名稱再根據型別查詢;
  • 支援的引數不同:@Autowired 只支援設定 1 個引數,而 @Resource 支援設定 7 個引數;
  • 依賴注入的用法支援不同:@Autowired 既支援構造方法注入,又支援屬性注入和 Setter 注入,而 @Resource 只支援屬性注入和 Setter 注入;
  • 編譯器 IDEA 的提示不同:當注入 Mapper 物件時,使用 @Autowired 註解編譯器會提示錯誤,而使用 @Resource 註解則不會提示錯誤。

以上就是一文搞懂Spring中@Autowired和@Resource的區別的詳細內容,更多關於Spring @Autowired @Resource區別的資料請關注it145.com其它相關文章!


IT145.com E-mail:sddin#qq.com