@Autowired, @Resource, @Inject 三个注解的区别你懂吗(别再乱用了!)

作者:Richard_Yi\
来源:juejin.cn/post/6844904056230690824
本章的内容主要是想探讨我们在进行Spring 开发过程当中,关于依赖注入的几个问题。感兴趣的读者可以先看下以下三点:

  • @Autowired, @Resource, @Inject 三个注解的区别
  • 当你在使用@Autowired时,是否有出现过Field injection is not recommended的警告?你知道这是为什么吗?
  • Spring 依赖注入有哪几种方式?官方是怎么建议使用的呢?
如果你对上述问题都了解,那我个人觉得你的开发经验应该是不错的。如果你看到这三个问题很懵可以往下看。
下面我们就依次对上述问题进行解答,并且总结知识点。
@Autowired, @Resource, @Inject 三个注解的区别 Spring 支持使用@Autowired, @Resource, @Inject 三个注解进行依赖注入。下面来介绍一下这三个注解有什么区别。
推荐一个 Spring Boot 基础教程及实战示例:
https://github.com/javastacks...
@Autowired
@Autowired为Spring 框架提供的注解,需要导入包org.springframework.beans.factory.annotation.Autowired
这里先给出一个示例代码,方便讲解说明:
public interface Svc {void sayHello(); }@Service public class SvcA implements Svc {@Override public void sayHello() { System.out.println("hello, this is service A"); }}@Service public class SvcB implements Svc {@Override public void sayHello() { System.out.println("hello, this is service B"); }}@Service public class SvcC implements Svc {@Override public void sayHello() { System.out.println("hello, this is service C"); } }

测试类:
@SpringBootTest public class SimpleTest {@Autowired // @Qualifier("svcA") Svc svc; @Test void rc() { Assertions.assertNotNull(svc); svc.sayHello(); }}

装配顺序:
1.按照type在上下文中查找匹配的bean,查找type为Svc的bean
2.如果有多个bean,则按照name进行匹配
  • 如果有@Qualifier注解,则按照@Qualifier指定的name进行匹配,查找name为svcA的bean
  • 如果没有,则按照变量名进行匹配,查找name为svcA的bean
3.匹配不到,则报错。(@Autowired(required=false),如果设置requiredfalse(默认为true),则注入失败时不会抛出异常)
@Inject
在Spring 的环境下,@Inject@Autowired 是相同的,因为它们的依赖注入都是使用AutowiredAnnotationBeanPostProcessor来处理的。
@Autowired, @Resource, @Inject 三个注解的区别你懂吗(别再乱用了!)
文章图片

@Inject是 JSR-330 定义的规范,如果使用这种方式,切换到Guice也是可以的。
Guice 是 google 开源的轻量级 DI 框架
如果硬要说两个的区别,首先@Inject是Java EE包里的,在SE环境需要单独引入。另一个区别在于@Autowired可以设置required=false@Inject并没有这个属性。
@Resource
@Resource是JSR-250定义的注解。Spring 在 CommonAnnotationBeanPostProcessor实现了对JSR-250的注解的处理,其中就包括@Resource
@Autowired, @Resource, @Inject 三个注解的区别你懂吗(别再乱用了!)
文章图片

@Resource有两个重要的属性:nametype,而Spring 将@Resource注解的name属性解析为bean的名字,而type属性则解析为bean的类型。
装配顺序:
  1. 如果同时指定了nametype,则从Spring上下文中找到唯一匹配的bean进行装配,找不到则抛出异常。
  2. 如果指定了name,则从上下文中查找名称(id)匹配的bean进行装配,找不到则抛出异常。
  3. 如果指定了type,则从上下文中找到类型匹配的唯一bean进行装配,找不到或是找到多个,都会抛出异常。
  4. 如果既没有指定name,又没有指定type,则默认按照byName方式进行装配;如果没有匹配,按照byTypeh进行装配。
近期热文推荐:
1.1,000+ 道 Java面试题及答案整理(2021最新版)
2.别在再满屏的 if/ else 了,试试策略模式,真香!!
3.卧槽!Java 中的 xx ≠ null 是什么新语法?
4.Spring Boot 2.5 重磅发布,黑暗模式太炸了!
5.《Java开发手册(嵩山版)》最新发布,速速下载!
【@Autowired, @Resource, @Inject 三个注解的区别你懂吗(别再乱用了!)】觉得不错,别忘了随手点赞+转发哦!

    推荐阅读