@Autowired和@Scope(BeanDefination.SCOPE_PROTOTYPE)区别

本文探讨了Spring框架中@Scope(BeanDefinition.SCOPE_PROTOTYPE)修饰的类在多线程环境下的最佳实践。当类被声明为原型作用域时,每次依赖注入都应产生新实例,推荐使用ApplicationContext.getBean()而非@Autowired。此外,文章还对比了@Resource与@Autowired的注解特性及装配顺序。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

当类被@Scope(BeanDefination.SCOPE_PROTOTYPE)修饰时,说明每次依赖注入时,都会产生新的对象,具体可参见文章:http://blog.youkuaiyun.com/gst6062825/article/details/8765157 

 

但是在多线程编程中,如果一个类被@Scope(BeanDefination.SCOPE_PROTOTYPE)修饰,且在另外的对象中如果需要引用此对象时,最好不要使用@Autowired依赖注入,而是使用 ApplicationContext.getBean()的方式去调用。

 

举例说明:

下面的这个类被@Scope(BeanDefinition.SCOPE_PROTOTYPE),


 
  1. @Component
  2. @Scope(BeanDefinition.SCOPE_PROTOTYPE)
  3. public class LoadData extends Thread {
  4. ...}

在MyMultiThread.java中需要多次使用上面的LoadData.java对象,每次都需要重新创建新的对象


 
  1. @Component
  2. public class MyMultiThread{
  3. .....
  4. for (多次) {
  5.          LoadData  loadData  =XXX.getApplicationContext().getBean( LoadData .class);
  6.   loadData  .start();
  7.     }
  8. }

通过上面的方式,使用getApplicationContext().getBean()能保证每次拿到的都是新创建的LoadData对象,而如果是在MyMultiThread中通过@Autowired去依赖注入,就不能(不会)每次都创建新的LoadData对象,除非是创建了多个

MyMultiThread对象,而这并不是我们希望做的。

 

总结:

当类被@Scope(BeanDefination.SCOPE_PROTOTYPE)修饰时,推荐使用getApplicationContext().getBean()去获取对象;

当类没有被@Scope(BeanDefination.SCOPE_PROTOTYPE)修饰时,可以使用@Autowired按类型自动注入,减少编码;

 

 

 


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方式进行装配;如果没有匹配,则回退为一个原始类型进行匹配,如果匹配则自动装配;

<think>首先,用户的问题是:“我想理解@Autowired@Resource注解在Java中的区别 请问Java中@Autowired@Resource注解的区别”。这基于系统级指令用户可见层的要求。 系统级指令: - 所有行内数学表达式必须使用$...$格式。但用户的问题是关于Java注解的,没有数学表达式,所以可能不需要。 - 独立公式用$$...$$,但同样,这里不涉及数学。 - LaTeX语法正确:确保如果使用数学公式,要正确。 - 使用中文回答:整个响应必须用中文。 - 生成相关问题:在回答后,添加相关问题部分。 - 引用标识:在回答中引用的段落末尾自然地添加引用标识,如[^1]。 用户可见层: - 回答结构清晰,帮助用户逐步解决问题。 - 尽量保证回答真实可靠。 - 参考站内引用:用户提供了引用[1]到[4],这些是关于@Autowired@Resource注解的区别的。我需要基于这些引用构建回答。 提供的引用: - [^1]: 讨论@Autowired@Resource的区别,包括来源不同(Spring vs Java规范)。 - [^2]: 强调来源不同依赖查找顺序。 - [^3]: 提到当接口有多个实现时,都需要名称匹配,@Autowired用@Qualifier,@Resource用name。 - [^4]: 给出示例代码,比较默认匹配策略。 回答结构: 1. **逐步解释**:清晰地解释每个注解,然后比较区别。 2. **使用引用**:在相关部分添加引用标识。 3. **语言**:全部用中文。 4. **相关问题**:在最后添加相关问题部分。 大纲回答: - **介绍**:简述两个注解的作用(用于依赖注入)。 - **来源不同**:@Autowired来自Spring框架,@Resource来自Java规范(JSR-250)。 - **依赖查找顺序**:解释默认行为的不同。 - **名称匹配**:当有多个实现时,如何指定名称。 - **示例**:提供代码示例来对比。 - **总结**:概括主要区别。 - **相关问题**:生成3-4个相关问题。 确保回答真实可靠:基于提供的引用。引用[1],[2],[3],[4]都一致地描述了这些区别。 最终响应结构: - 回答部分:用清晰段落。 - 相关问题部分:以“§§相关问题§§”开头,列出问题。 现在,构建回答。</think>### Java中@Autowired@Resource注解的区别 在Java开发中,特别是使用Spring框架时,`@Autowired``@Resource`都是常用的注解,用于实现依赖注入(DI)。它们的主要目的是自动装配Bean,减少手动配置的代码量。然而,它们在来源、依赖查找顺序、名称匹配策略等方面存在显著差异。下面我将逐步解释这些区别,帮助您清晰理解。 #### 1. **来源不同** - **@Autowired**:这是Spring框架专有的注解,由Spring 2.5版本引入。它属于Spring的生态系统,主要用于Spring容器中的Bean管理[^2]。 - **@Resource**:这是Java标准规范的一部分,源自JSR-250(Java Specification Request 250)。它不依赖于Spring,而是Java EE/Jakarta EE的一部分,因此更通用,可以在非Spring环境中使用[^1][^2]。 #### 2. **依赖查找顺序不同** - **@Autowired**:默认按**类型(byType)** 进行依赖查找。也就是说,Spring容器会查找与字段或方法参数类型匹配的Bean。如果找到多个匹配的Bean,它会抛出异常,除非指定了名称[^2][^4]。 - 例如:`@Autowired private NotificationService service;` 会注入类型为`NotificationService`的Bean。 - **@Resource**:默认按**名称(byName)** 进行依赖查找。它会先根据Bean的名称匹配,如果名称未指定,再回退到按类型匹配。这减少了歧义,尤其在多个实现存在时[^1][^4]。 - 例如:`@Resource private NotificationService service;` 会查找名为"service"的Bean。 #### 3. **名称匹配策略(当接口有多个实现时)** 当同一个接口有多个实现类时,两者都需要指定Bean名称来避免歧义,但指定方式不同: - **@Autowired**:需要配合`@Qualifier`注解来显式指定Bean名称。这增加了灵活性,但代码稍显冗长[^3][^4]。 - 示例代码: ```java @Autowired @Qualifier("emailService") private NotificationService service; ``` - **@Resource**:可以直接使用`name`属性指定Bean名称,不需要额外注解。这更简洁,且符合Java标准[^3][^4]。 - 示例代码: ```java @Resource(name = "emailService") private NotificationService service; ``` #### 4. **默认匹配策略兼容性** - **@Autowired**:作为Spring特有注解,它深度集成Spring的功能,如支持构造函数注入字段注入。但它不适用于非Spring项目[^4]。 - **@Resource**:作为Java标准注解,它兼容性更好,可以在任何支持JSR-250的容器中使用(如Tomcat或WildFly)。但它在Spring中可能不如`@Autowired`灵活,尤其在处理复杂依赖时[^1][^2]。 #### 总结 - **主要区别**:`@Autowired`是Spring专有,默认按类型查找,需用`@Qualifier`指定名称;`@Resource`是Java标准,默认按名称查找,可直接用`name`属性指定名称[^1][^2][^3][^4]。 - **推荐使用场景**: - 在纯Spring项目中,`@Autowired`更常用,因为它与Spring特性无缝集成。 - 在需要跨框架兼容性或避免Spring依赖时,优先使用`@Resource`。 - **注意事项**:两者都能简化依赖注入,但过度使用可能导致代码耦合。建议结合单元测试确保注入正确。 如果您有具体代码场景,我可以进一步分析!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值