@Autowired為什么會報錯?如何解決?
@Autowired報錯信息相信大部分程序員都遇到過,奇怪的是雖然代碼報錯,但絲毫不影響程序的正常執(zhí)行,也就是雖然編譯器 IDEA 報錯,但程序卻能正常的執(zhí)行,那這其中的原因又是為何?
報錯原因分析
報錯的原因首先是因為 IDEA 強大的報警機制,@Autowired 為 Spring 的注解,含義是將某類動態(tài)的注入到當前類中,如下圖所示:
@Autowired 默認是根據(jù) type 進行注入,并且注入時要求(注入)對象不能為 NULL,默認值如下圖所示:
而 IDEA 報錯的原因是:@Autowired 為 Spring 的注解,而注入的 Mapper 對象使用的又是 @Mapper 的注解,然而 @Mapper 又為 MyBaits 的注解,IDEA 能很好的兼容并識別 Spring 的注解,但不能很好的識別 MyBatis 的注解,因此在使用 @Autowired 注解時,IDEA 并不能檢測到 @Mapper 注解的對象不為 NULL,因此就會報錯。
這就是為什么使用 Spring 的注解 @Repository/@Component... 不報錯,而使用 @Mapper 注解卻會報錯的根本原因,如下圖所示:
解決方案1:關閉報警機制
關閉 IDEA 注入報警機制,可以避免報錯,實現(xiàn)步驟如下。
打開 IDEA,找到參數(shù)設置選項 “Preferences...” ,如下圖所示:
依次選擇 “Editor” -> “Inspections” -> “Spring” -> “Spring Core” -> “Code” -> “Autowiring for bean class” 將 “Error” 級別修改成 “Waring” 級別,如下圖所示:
設置完成之后點擊確認,查看之前報錯的 Mapper 類,此時展示效果如下:
報錯信息消失了。
解決方案2:添加Spring注解
在 Mapper 的類上添加 Spring 的注解,也可以解決 IDEA 報錯的問題,如 @Repository 或 @Component 這類注解,如下圖所示:
或使用 @Repository 注解,如下圖所示:
查看之前的報錯信息:
報錯消失了。
解決方案3:允許注入對象為NULL
設置允許注入的 Mapper 對象為 NULL,也可以避免 IDEA 報錯,只需要設置 @Autowired(required=false) 即可,如下圖所示:
(其中 userMapper2 對象就不報錯了)
- @Autowired(required=true):表示當使用 @Autowired 注解的時候,該 bean 必須存在,否則注入失敗,默認值。
- @Autowired(required=false):表示忽略當前要注入的 bean,如果有直接注入,沒有則跳過,不會報錯。
@Autowired 默認值的實現(xiàn)源碼:
解決方案4:使用@Resource注解
使用 @Resource 注解替換 @Autowired 注解也可以避免報錯,它們的對比效果如下:
@Resource 注解和 @Autowired 注解以當前的場景來說,它們的主要區(qū)別是 @Resource 是 Java 自身提供的注解,而 @Autowired 是 Spring 提供的注解,@Autowired 默認值為 required=true,所以必須要一個非 NULL 的對象,當 IDEA 檢測不到對象為 NULL 時就會報錯,而 @Resource 并沒有這項要求。
小結
使用 @Autowired 注解導入 Mapper 對象報錯的原因,是因為 @Autowired 默認情況下,需要注入一個非 NULL 的對象,而被 @Mapper 修飾的類為 MyBatis 的注解,IDEA 并不能很好的識別其為非 NULL 對象,因此就會報錯。當然,它的解決方案也有很多,推薦使用 @Resource 替代 @Autowired 注解的方式來解決此問題。