本章的内容主要是想探讨我们在进行Spring 开发过程当中,关于依赖注入的几个问题。感兴趣的读者可以先看下以下三点:
@Autowired, @Resource, @Inject 三个注解的区别
当你在使用@Autowired时,是否有出现过Field injection is not recommended的警告?你知道这是为什么吗?
Spring 依赖注入有哪几种方式?官方是怎么建议使用的呢?
如果你对上述问题都了解,那我个人觉得你的开发经验应该是不错的👍。如果你看到这三个问题很懵可以往下看。
下面我们就依次对上述问题进行解答,并且总结知识点。
@Autowired, @Resource, @Inject 三个注解的区别
Spring 支持使用@Autowired, @Resource, @Inject 三个注解进行依赖注入。下面来介绍一下这三个注解有什么区别。
推荐一个 Spring Boot 基础教程及实战示例: https://github.com/javastacks/javastack
@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),如果设置required为false(默认为true),则注入失败时不会抛出异常)
@Inject
在Spring 的环境下,@Inject和@Autowired 是相同的,因为它们的依赖注入都是使用AutowiredAnnotationBeanPostProcessor来处理的。@Inject是 JSR-330 定义的规范,如果使用这种方式,切换到Guice也是可以的。
Guice 是 google 开源的轻量级 DI 框架
如果硬要说两个的区别,首先@Inject是Java EE包里的,在SE环境需要单独引入。另一个区别在于@Autowired可以设置required=false而@Inject并没有这个属性。
@Resource
@Resource是JSR-250定义的注解。Spring 在 CommonAnnotationBeanPostProcessor实现了对JSR-250的注解的处理,其中就包括@Resource。
@Resource有两个重要的属性:name和type,而Spring 将@Resource注解的name属性解析为bean的名字,而type属性则解析为bean的类型。
装配顺序:
如果同时指定了name和type,则从Spring上下文中找到唯一匹配的bean进行装配,找不到则抛出异常。
如果指定了name,则从上下文中查找名称(id)匹配的bean进行装配,找不到则抛出异常。
如果指定了type,则从上下文中找到类型匹配的唯一bean进行装配,找不到或是找到多个,都会抛出异常。
如果既没有指定name,又没有指定type,则默认按照byName方式进行装配;如果没有匹配,按照byTypeh进行装配。