Spring中的@Autowire、@Resource、@Qualifier的区别

举例说明:

  1. 接口:IAnimal
public Interface IAnimal{
    ......
}
  1. 实现类:DogImpl ,实现了IAnimal接口。
@Service("dogImpl")
public class DaoImpl impliments IAnimal{
    ...
}
  1. 业务类:AnimalController
public class AnimalController {
    @Autowired
    private IAnimal dogImpl;
    ......
}

假如有一个“动物”的接口 IAnimal, DogImpl类实现了接口 IAnimal, 且该接口只有 DogImpl这一个实现类,那么在引用实现类的时候,我们使用的是实现类的接口(像上面程序展示的那样)。Spring会按 byType的方式寻找接口的实现类,将其注入。
假如有另一个实现类 CatImpl 也实现了接口 IAnimal, 这时候再按上面的方式去引用, 在同时存在两个实现类的情况下,会出现什么情况呢?
答:会报错。 这是由于 @Autowired 的特性决定的: @Autowired 的注入方式是 byType 注入, 当要注入的类型在容器中存在多个时,Spring是不知道要引入哪个实现类的,所以会报错。
那么在同一类型拥有多个实现类的时候,如何注入呢?
答:这种场景下,只能通过 byName 注入的方式。可以使用 @Resource 或 @Qualifier 注解。

@Resource 默认是按照 byName 的方式注入的, 如果通过 byName 的方式匹配不到,再按 byType 的方式去匹配。所以上面的引用可以替换为:

public class AnimalController {
    @Resource(name="dogImpl")        //实现类1中 @Service注解中标定的名称
    private IAnimal dogImpl;
    ......
}
       @Qualifier 注解也是 byName的方式,但是与@Resource 有区别,@Qualifier 使用的是 类名。
public class AnimalController {
    @Qualifier("DaoImpl")        //实现类1的类名。注意区分与@Resource(name="dogImpl") 的区别。
    private IAnimal dogImpl;
    ......
}

@Resource 相当与 @Autowire 加上 @Qualifier
@Autowire默认按照类型装配,默认情况下必须要求依赖对象必须存在,如果要允许null值,可以设置它的required属性为false。
@Resource装配顺序
如果同时指定name和type,则从容器中查找唯一匹配的bean装配,找不到则抛出异常
如果指定name属性,则从容器中查找名称匹配的bean装配,找不到则抛出异常
如果指定type属性,则从容器中查找类型唯一匹配的bean装配,找不到或者找到多个抛出异常
如果都不指定,则自动按照byName方式装配,如果没有匹配,则回退一个原始类型进行匹配,如果匹配则自动装配

总结


1、@Autowired 是通过 byType 的方式去注入的, 使用该注解,要求接口只能有一个实现类。
2、@Resource 可以通过 byName 和 byType的方式注入, 默认先按 byName的方式进行匹配,如果匹配不到,再按 byType的方式进行匹配。
3、@Qualifier 注解可以按名称注入, 但是注意是 类名。

你可能感兴趣的:(Spring中的@Autowire、@Resource、@Qualifier的区别)