@Autowired和@Resource的区别

Posted by zengchengjie on Tuesday, February 11, 2020

概念

点击参考原文章

@Resource和@Autowired都是做bean的注入时使用,其实@Resource并不是Spring的注解,它的包是javax.annotation.Resource,需要导入,但是Spring支持该注解的注入。

1、共同点

两者都可以写在字段和setter方法上。两者如果都写在字段上,那么就不需要再写setter方法。

2、不同点

(1)@Autowired

@Autowired为Spring提供的注解,需要导入包org.springframework.beans.factory.annotation.Autowired;只按照byType注入。

public class TestServiceImpl {
    // 下面两种@Autowired只要使用一种即可
    @Autowired
    private UserDao userDao; // 用于字段上
    
    @Autowired
    public void setUserDao(UserDao userDao) { // 用于属性的方法上
        this.userDao = userDao;
    }
}

@Autowired注解是按照类型(byType)装配依赖对象,默认情况下它要求依赖对象必须存在,如果允许null值,可以设置它的required属性为false。如果我们想使用按照名称(byName)来装配,可以结合@Qualifier注解一起使用。(通过类型匹配找到多个candidate,在没有@Qualifier、@Primary注解的情况下,会使用对象名作为最后的fallback匹配)如下:

public class TestServiceImpl {
    @Autowired
    @Qualifier("userDao")
    private UserDao userDao; 
}

(2)@Resource

@Resource默认按照ByName自动注入,由J2EE提供,需要导入包javax.annotation.Resource。@Resource有两个重要的属性:name和type,而Spring将@Resource注解的name属性解析为bean的名字,而type属性则解析为bean的类型。所以,如果使用name属性,则使用byName的自动注入策略,而使用type属性时则使用byType自动注入策略。如果既不制定name也不制定type属性,这时将通过反射机制使用byName自动注入策略。

public class TestServiceImpl {
    // 下面两种@Resource只要使用一种即可
    @Resource(name="userDao")
    private UserDao userDao; // 用于字段上
    
    @Resource(name="userDao")
    public void setUserDao(UserDao userDao) { // 用于属性的setter方法上
        this.userDao = userDao;
    }
}

注:最好是将@Resource放在setter方法上,因为这样更符合面向对象的思想,通过set、get去操作属性,而不是直接去操作属性。

@Resource装配顺序:

①如果同时指定了name和type,则从Spring上下文中找到唯一匹配的bean进行装配,找不到则抛出异常。

②如果指定了name,则从上下文中查找名称(id)匹配的bean进行装配,找不到则抛出异常。

③如果指定了type,则从上下文中找到类似匹配的唯一bean进行装配,找不到或是找到多个,都会抛出异常。

④如果既没有指定name,又没有指定type,则自动按照byName方式进行装配;如果没有匹配,则回退为一个原始类型进行匹配,如果匹配则自动装配。

@Resource的作用相当于@Autowired,只不过@Autowired按照byType自动注入。

我碰到的问题

一般来说,多个bean会出现类似的冲突问题

第一个bean

@Configuration
public class FirstBean {
    public static final String FIRST_BEAN = "first";
    @Bean(name = FIRST_BEAN)
    public BeanModel beanModel(){
        return new BeanModel(1);
    }
}

第二个bean:

@Configuration
public class SecondBean {
    public static final String SECOND_BEAN = "second";
    @Bean(name = SECOND_BEAN)
    public BeanModel beanModel(){
        return new BeanModel(2);
    }
}

bean model

public class BeanModel {
    int type;
    public BeanModel(int type) {
        System.out.println("bean:"+type);
        this.type = type;
    }
    public void printType(){
        System.out.println("type:"+type);
    }
}

引用这个bean的地方:

@Configuration
public class BeanTestConfig {
    @Resource(name = FIRST_BEAN)
    private BeanModel firstBeanTest;
    //注入bean需要使用resource或者autowired注解,否则报空指针错误
    //正确用法1:使用resource注解并指定name
    @Resource(name = SECOND_BEAN)
//    如果使用resource注解,不指定name会报available: expected single matching bean but found 2: first,second
//    错误:找到两个bean(需要你指定具体哪个)
//    @Resource
//    正确用法2:使用autowired 需要配合Qualifier来指定name使用
//    @Autowired
//    @Qualifier(SECOND_BEAN)
    private BeanModel secondBeanTest;

    @Bean
    public void printBean(){
        firstBeanTest.printType();
        secondBeanTest.printType();
    }
}

inject注解

**@注入:**基于 JSR-330(Java 依赖注入)的注解标识可注入的构造函数、方法和字段。这个注解几乎完全替代了 Spring 的 @Autowired 注解。因此,您可以选择使用 @Inject,而不是使用 Spring 特定的 @Autowired 注解。@Autowired 和 @Inject 之间的区别之一是 @Inject 没有必填字段,因此如果我们无法找到合适的注入对象,它将失败,而 @Autowired 可以使用 required=false 并允许可以为空的字段(仅如果需要!)。@Inject 注解的优点是,你可以让@Inject 注入一个Provider,而不是直接注入一个引用。Provider 接口可以实现 bean 引用的惰性注入和 bean 的多个实例的注入。如果我们很少实现接口或子类,我们可以使用 @Named 注释来缩小选择范围以避免歧义。@Named 注解的工作方式很像 Spring 的 @Qualifier