1)@Resource (默认首先按名称匹配注入,或者排除

作者:美狮美高梅官方网站

图片 1image

  • @Resource、@Autowired、@Qualifier 区别(表格显示)

具有依赖关系的Bean对象,利用下面任意一种注解都可以实现关系注入:
1)@Resource (默认首先按名称匹配注入,然后类型匹配注入)
2)@Autowired/@Qualifier (默认按类型匹配注入)
3)@Inject/@Named

  • Spring通过注解使用多数据源
区别项 @Resource @Autowired @Qualifier
谁提供的 jdk提供,包是javax.annotation.Resource Spring提供,包是org.springframework.beans.factory.annotation.Autowired jdk提供,包是javax.inject.Qualifier
注入方式 可以以 name 或 type 方式注入 只能以 type 方式注入 只能以 name 方式注入
默认注入方式 默认以 name 方式注入
搭配使用 可以搭配 @Autowired 使用,区别多个具有相同类型的 bean

注意:这三个没有任何区别,无任何限制,只是第三个要额外导包。

坑:@Autowired 按 byType 自动注入,而 @Resource 则默认按 byName 自动注入,@Primary是优先选择。

  • 问:如果在容器中存在多个相同类型的bean,怎么办呢?
    答:spring提供了另外两种选择,可以设置一个首选bean,或者排除一些bean。
    还有,@Autowired 和 @Qualifier 搭配

  • byName就是通过Bean的id或者name,byType就是按Bean的Class的类型。

  • 如果是属性注入,需要为每一个依赖类创建相应的getter和setter方法。
    如果是构造方法注入,需要为依赖类创建相应的构造方法。

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

对前两个进行区别如下:

例如,在项目中是有两个Redis源,这两个Redis Bean分别为dataRedisTemplate和redisTemplate。

  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注解一起使用。如下:

    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去操作属性,而不是直接去操作属性。

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

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

(1)@Resource(这个注解属于J2EE的)

Redis Bean1:dataRedisTemplate,clusterNodes为${data-redis.cluster.nodes}

 @Resource装配顺序

@Bean(name = "dataRedisTemplate")public RedisTemplate dataRedisTemplate() { RedisTemplate template = new RedisTemplate(); template.setConnectionFactory(sessionLettuceConnectionFactory); Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class); ObjectMapper om = new ObjectMapper(); om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY); om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL); jackson2JsonRedisSerializer.setObjectMapper; template.setKeySerializer(new StringRedisSerializer; template.setValueSerializer(jackson2JsonRedisSerializer); template.setHashKeySerializer(jackson2JsonRedisSerializer); template.setHashValueSerializer(new StringRedisSerializer; template.afterPropertiesSet(); return template;}// factory@Resource@Qualifier(value = "dataLettuceConnectionFactory")private RedisConnectionFactory dataLettuceConnectionFactory;// clusterNodes@Value("${spring.data-redis.cluster.nodes}")private String clusterNodes;
  1. 如果同时指定了name和type,则从Spring上下文中找到唯一匹配的bean进行装配,找不到则抛出异常
  2. 如果指定了name,则从上下文中查找名称(id)匹配的bean进行装配,找不到则抛出异常
  3. 如果指定了type,则从上下文中找到类型匹配的唯一bean进行装配,找不到或者找到多个,都会抛出异常
  4. 如果既没有指定name,又没有指定type,则自动按照byName方式进行装配;如果没有匹配,则回退为按类型进行匹配,如果匹配则自动装配;

Redis Bean2:redisTemplate,clusterNodes为${redis.cluster.nodes}

来看一个使用 @Resource 的例子:

@Primary@Bean(name = "redisTemplate")public RedisTemplate<String, Object> redisTemplate() { RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>(); redisTemplate.setConnectionFactory(lettuceConnectionFactory); redisTemplate.setKeySerializer(new StringRedisSerializer; Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class); ObjectMapper objectMapper = new ObjectMapper(); objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY); objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL); jackson2JsonRedisSerializer.setObjectMapper(objectMapper); redisTemplate.setValueSerializer(jackson2JsonRedisSerializer); redisTemplate.setHashKeySerializer(new StringRedisSerializer; redisTemplate.setHashValueSerializer(new StringRedisSerializer; redisTemplate.afterPropertiesSet(); return redisTemplate;}// factory@Resource@Qualifier(value = "lettuceConnectionFactory")private RedisConnectionFactory lettuceConnectionFactory;// clusterNodes@Value("${spring.redis.cluster.nodes}")private String clusterNodes;

使用 @Resource 注释的 Test.java

在一个应用中要把数据放入到“Redis Bean1:dataRedisTemplate”对应的Redis中,于是我在这个应用中使用方式如下:

package com.test;

@Autowiredprivate RedisTemplate dataRedisTemplate;// 根据key获取数据Object obj = dataRedisTemplate.opsForValue().get;

import javax.annotation.Resource;

实际上使用的是“Redis Bean2:redisTemplate”对应的Redis。破解方式一:把@Autowired换成@Resource 注解。如下:

public class Test{

@Autowiredprivate RedisTemplate dataRedisTemplate;// 把@Autowired换成@Resource@Resourceprivate RedisTemplate dataRedisTemplate;

  @Resource

@Autowired和@Resource最大的区别就是:@Autowired 按 byType 自动注入,而 @Resource 则默认按 byName 自动注入。

  private Car car;

本文由美狮美高梅官方网站发布,转载请注明来源

关键词: