Spring注解下的@Resource和@Autowried的区别

目录

@Autowired 与@Resource解释:

@Autowired 与@Resource的区别:

@Autowired注解可能出现的问题及解决方法:

①存在多个BaseService实例的情况:

②没有BaseService实例的情况:

总结:


@Autowired 与@Resource解释:

Spring不但支持自己定义的@Autowired注解,还支持几个由JSR-250规范定义的注解,它们分别是@Resource、@PostConstruct以及@PreDestroy。
  @Resource的作用相当于@Autowired,只不过@Autowired按byType自动注入,而@Resource默认按 byName自动注入罢了。@Resource有两个属性是比较重要的,分是name和type,Spring将@Resource注解的name属性解析为bean的名字,而type属性则解析为bean的类型。所以如果使用name属性,则使用byName的自动注入策略,而使用type属性时则使用byType自动注入策略。如果既不指定name也不指定type属性,这时将通过反射机制使用byName自动注入策略。
  @Resource装配顺序
  1. 如果同时指定了name和type,则从Spring上下文中找到唯一匹配的bean进行装配,找不到则抛出异常
  2. 如果指定了name,则从上下文中查找名称(id)匹配的bean进行装配,找不到则抛出异常
  3. 如果指定了type,则从上下文中找到类型匹配的唯一bean进行装配,找不到或者找到多个,都会抛出异常
  4. 如果既没有指定name,又没有指定type,则自动按照byName方式进行装配;如果没有匹配,则回退为一个原始类型进行匹配,如果匹配则自动装配。

@Autowired 与@Resource的区别:

1、 @Autowired与@Resource都可以用来装配bean. 都可以写在字段上,或写在setter方法上。

2、 @Autowired默认按类型装配(这个注解是属于spring的),默认情况下必须要求依赖对象必须存在,如果要允许null值,可以设置它的required属性为false,如:@Autowired(required=false) ,如果我们想使用名称装配可以结合@Qualifier注解进行使用,如下:

@Autowired()

@Qualifier("baseService")

private BaseService baseService;

3、@Resource(这个注解属于J2EE的),默认按照名称进行装配,名称可以通过name属性进行指定,如果没有指定name属性,当注解写在字段上时,默认取字段名进行安装名称查找,如果注解写在setter方法上默认取属性名进行装配。当找不到与名称匹配的bean时才按照类型进行装配。但是需要注意的是,如果name属性一旦指定,就只会按照名称进行装配

@Resource(name="baseService")

private BaseService baseService;

@Autowired注解可能出现的问题及解决方法:

@Autowired是根据类型进行自动装配的。如果当Spring上下文中存在不止一个BaseService类型的bean时,就会抛出BeanCreationException异常;如果Spring上下文中不存在BaseService类型的bean,也会抛出BeanCreationException异常。我们可以使用@Qualifier配合@Autowired来解决这些问题。

①存在多个BaseService实例的情况:

假设我们如下的一个接口和两个实现类:

interface BaseService{}

class BaseService1 implments BaseService{}

class BaseService2 implments BaseService{}

此时我们在service中使用@Autowired注解进行注入的时候:

@RestController
public class Controller {
    @Autowired
    private BaseService baseService;

    @RequestMapping("/query")
    public String query() { return "hello";}

此时运行项目的时候则会报: 

Description:

Field baseService in com.test.Controller required a single bean, but 2 were found:
    - baseService1: defined in file [com\test\service\impl\BaseService1.class]
    - baseService2: defined in file [com\test\service\impl\BaseService2.class]
Action:

Consider marking one of the beans as @Primary, updating the consumer to accept multiple beans, or using @Qualifier to identify the bean that should be consumed
 

我们则需要更改为:

@RestController
public class Controller {

    @Autowired
    @Qualifier("baseService1")
    private BaseService baseService;

    @RequestMapping("/query") 
    public String query() { return "hello"; }
}

因为@Autowired注解是根据byType类型的,我们注入了BaseDao,但是其实项目里有BaseDao1和BaseDao2两个实现类,程序便会报错,当使用 @Qualifier("baseService1")时候便会明确了使用哪个BaseDao的具体实现类。

②没有BaseService实例的情况:

假设我们只有一个接口:

interface BaseService{}

此时我们注入使用:

@RestController
public class Controller {
    @Autowired
    private BaseService baseService;

    @RequestMapping("/query")
    public String query() { return "hello"; }
}

运行项目的时候则会报: 

Description:

Field baseService in com.test.Controller required a bean of type 'com.test.service.BaseService' that could not be found.
Action:

Consider defining a bean of type 'com.test.service.BaseService' in your configuration.

将代码改为:

@RestController
public class Controller {
    //required = false 这种方式不推荐,将问题抛在启动时候检测而不是将问题留在运行的时候在抛出
    @Autowired(required = false)
    private BaseService baseService;

    @RequestMapping("/query")
    public String query() { return "hello"; }
}

总结:

@Autowired默认是byType

@Resource默认是byName

  • 3
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值