Spring中@Resource和@Autowire的区别

1、实践出真知

@Resource 是 JSR-250 规范的一部分

说到 @Resource 大家肯定会想到 @Autowired, 至于两者的区别, 大部门分童鞋都知道。


@Resource 的作用和 @Autowired 一样,只不过 @Autowired 是按 byType 自动注入,而 @Resource 默认按 byName 自动注入,而且还提供了 name 和 type 两个属性,其含义也容易理解,分别按 byName 和 byType 注入。

如果两个属性都没有提供的话,则根据属性名称注入,我理解的是 byName,还有地方说是通过反射机制使用 byName自动注入。

俗话说的好 no tu no bb,接下来多图慎入。下图为测试项目的结构。

 

(1)@Autowired,匹配多个实现类,但默认属性名匹配不上,报错

Spring中@Resource和@Autowire的区别_第1张图片

我们先来看看 @Autowired 的测试情况。

Spring中@Resource和@Autowire的区别_第2张图片

Spring中@Resource和@Autowire的区别_第3张图片

我通过在 SpringTest 类的 Main 方法中,初始化和启动 Spring 容器。配置中配置了两个 MyTestService 接口的实现类 MyTestServiceImpl2 和 MyTestServiceImpl3,打印了简单的一句话。结果肯定显而易见,报错了。

Spring中@Resource和@Autowire的区别_第4张图片

我在 注解@Service 中配置了两个实现类,默认按 byType 注入,发现多个,根据 byName注入,无法找到 name 为 myTestService 的 bean,所以报错了,接下来修改属性名,再看。

 

(2)@Autowired,匹配多个实现类,默认属性名匹配上其中一个,正常

这次同样的步骤,默认按 byType 注入,发现多个,根据 byName 注入,找到 name 为 myTestServiceImpl2 的 Bean。成功执行。

 

(3)@Resource 没指定name属性,且默认属性名匹配不上,但只匹配上一个实现类,正常

来看看 @Resource 的情况,请看下图配置,我注释掉了 MyTestServiceImpl2 实现类的声明。

Spring中@Resource和@Autowire的区别_第5张图片

Spring中@Resource和@Autowire的区别_第6张图片

注意 HomeController 中声明的属性 myTestService,在配置中没有这个属性对应的 Bean,@Resource 没有配置 name 和 type 的情况下,结果是什么呢?现在大家猜一下执行成功还是失败。

是的,没有报错,你猜对了吗?那好现在我们把 HomeController 中的属性名称再改一下。

 

(4)@Resource 没指定name属性,但默认属性名匹配,正常

我把属性改为存在的 Bean,myTestServiceImpl3,然后我们再来看看结果,毫无疑问结果肯定是没有报错。

那好,按照以上测试结果来看,当 @Resource 没有提供 name 和 type 属性的时候,如果 byName 没有找到对应的 Bean 时,则会根据依赖属性的类型去 Spring 容器中查找是否有提供了其他类型相同的 Bean,如果有则自动注入,如果没有则报错。当然如果根据 byName 找到了则直接注入。接下来,我们再来看看,name 和 type 的情况。

 

(5)@Resource 指定name属性,但属性名不匹配,报错

我在 HomeController 中添加了 @Resource 的 name 属性 为 myTestServiceImpl2,那么现在如果执行,结果是否会报错?

Spring中@Resource和@Autowire的区别_第7张图片

是的报错了,因为注释了 myTestServiceImpl2 的声明,所以在使用 name 属性查找时,无法找到对应的 Bean,所以直接报错,没有再次去通过类型到 Spring 容器中查找。接下来我们再看看使用配置的 Bean的情况。

 

(6)@Resource 指定name属性,属性名匹配,正常

我使用了 userDaoImpl1 这个 Bean,那结果显而易见,肯定没有问题。

type 属性我就不用演示了吧,和 name 属性是一样,如果同时提供了 name 和 type 的话,那就必须两个都要匹配才能注入依赖,否则就会注入失败。总算把 @Resource 注解搞明白了,之前一直以为 @Resource 注解就是默认 byName 装配,不知道里面还有这么多弯弯绕呢。

 

2、@Autowired

@Autowired默认byType自动注入,如果实例结果不唯一,那么将会抛出异常

@Autowired可与@Qualifier("beanName")搭配使用,注入指定bean。

@Autowired
@Qualifier("baseDao")
private BaseDao baseDao;

如,同一个接口,两个实现类,就可以使用该方式指定注入。

@Autowired(required=false)表示如果spring上下文中没有找到该类型的bean ,将会使用

new SoftPMServiceImpl();

private ISoftPMService softPMService = new SoftPMServiceImpl();

 

3、@Resource

@Resource默认按byName自动注入
但是@Resource有两个属性是比较重要的,分别是name和type;
如果使用name属性,则使用byName的自动注入策略; .
而使用type属性时则使用byType自动注入策略;

@Resource装配顺序
  1. 如果同时指定了name和type,则从Spring上下文中找到唯一匹配的bean进行装配,找不到则抛出异常
  2. 如果指定了name,则从上下文中查找名称(id)匹配的bean进行装配,找不到则抛出异常
  3. 如果指定了type,则从上下文中找到类型匹配的唯一bean进行装配,找不到或者找到多个,都会抛出异常
  4. 如果既没有指定name,又没有指定type,则自动按照byName方式进行装配;如果没有匹配,则回退为一个原始类型进行匹配,如果匹配则自动装配;

使用方式如下:

//@Resource(name="dataSource");

@Resource(type="DataSource.class");

@Resource
private DataSource dataSource;
//inject the bean named 'dataSource';

 

4、简要对比表格

注解对比 @Resource @Autowire
注解来源 JDK Spring
装配方式 优先按名称 优先按类型
属性 name、type required


参考:

https://zhuanlan.zhihu.com/p/114376607

你可能感兴趣的:(Spring,MVC)