51工具盒子

依楼听风雨
笑看云卷云舒,淡观潮起潮落

@AutoWired和 @Resource原理分析!

嗨,你好呀,我是猿java

@Autowired@Resource是 Java程序员经常用来实现依赖注入的两个注解,这篇文章,我们将详细分析这两个注解的工作原理、使用示例和它们之间的对比。

依赖注入概述 {#依赖注入概述}

依赖注入是一种常见的设计模式,用于实现控制反转(Inversion of Control, IoC)。在传统的编程中,类通常负责管理自己的依赖,而在 DI中,这种责任被转移到了外部容器(如 Spring容器)上,通过 DI,可以提高代码的可测试性和可维护性,因为依赖关系是通过配置而不是硬编码的。

@Autowired {#Autowired}

@Autowired是 Spring框架提供的注解,用于自动装配 bean。它可以用在构造器、方法、字段或参数上,Spring容器会通过类型匹配(byType)来注入依赖。

@Autowired的工作原理 {#Autowired的工作原理}

  • 类型匹配:Spring首先通过类型匹配来查找合适的bean。如果找到一个唯一的bean,则注入该bean。
  • 候选Bean的歧义:如果有多个同类型的bean,Spring会通过字段名或参数名来进一步匹配。
  • @Primary注解 :可以使用@Primary注解来标记一个bean为主要候选者。
  • @Qualifier注解 :在多个候选bean的情况下,可以使用@Qualifier注解来指定注入的bean。

作用范围 {#作用范围}

@Autowired的作用范围包括三种:

  • 字段注入:最简单的方式,但不利于单元测试,因为依赖是通过反射注入的。
  • 构造器注入:推荐的方式,因为它可以确保依赖在对象创建时就被注入。
  • 方法注入:通过一个setter方法注入依赖。

在Spring框架中,依赖注入可以通过多种方式来实现,主要包括构造器注入、字段注入和方法注入。每种方式都有其特定的使用场景和优缺点。下面我将为每种注入方式提供示例代码,以帮助理解其实现和适用场景。

字段注入 {#字段注入}

字段注入是通过直接在类的字段上使用注解来实现的,这种方式最为简单,但也有一些缺点,特别是在单元测试中,因为它依赖于反射来设置字段的值。Spring官方已经不建议这种使用方式。

如下示例展示了字段注入:

|---------------------------------------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 | import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; @Component public class CarService { @Autowired private Engine engine; public void start() { engine.run(); } } @Component public class Engine { public void run() { System.out.println("Engine is running"); } } |

适用场景 {#适用场景}

  • 快速原型:在快速开发或原型阶段,字段注入可以减少样板代码。
  • 简单的依赖关系:如果类的依赖关系简单且不需要复杂的初始化逻辑,字段注入可以提供一种直接的方式。

缺点 {#缺点}

  • 测试困难:由于字段是私有的,单元测试时需要使用反射来设置字段的值,这增加了复杂性。
  • 不支持final字段:因为字段注入发生在对象实例化之后,无法用于final字段。

构造器注入 {#构造器注入}

构造器注入是通过类的构造函数来实现依赖注入的,这种方式被广泛推荐,因为它可以在对象创建时确保所有依赖都被正确注入,从而避免未初始化的依赖。

如下示例展示了构造器注入:

|---------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 | import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; @Component public class CarService { private final Engine engine; @Autowired // 可以省略,因为只有一个构造函数时,Spring会自动注入 public CarService(Engine engine) { this.engine = engine; } public void start() { engine.run(); } } @Component public class Engine { public void run() { System.out.println("Engine is running"); } } |

适用场景 {#适用场景-1}

  • 不可变性:如果你希望你的类是不可变的,构造器注入是最佳选择,因为它可以确保所有依赖在对象创建时都被注入。
  • 必需依赖:如果某个依赖是必需的,构造器注入可以确保在对象创建时注入该依赖。

方法注入 {#方法注入}

方法注入(通常是 setter方法注入)是通过提供一个公共的 setter方法来实现的。这种方式提供了一种在对象实例化后设置依赖的灵活性。

如下示例展示了方法注入:

|---------------------------------------------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 | import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; @Component public class CarService { private Engine engine; @Autowired public void setEngine(Engine engine) { this.engine = engine; } public void start() { engine.run(); } } @Component public class Engine { public void run() { System.out.println("Engine is running"); } } |

适用场景 {#适用场景-2}

  • 可选依赖:如果某个依赖是可选的,方法注入可以让你在必要时设置该依赖。
  • 需要后期配置:如果需要在对象实例化后进行额外的配置或初始化,方法注入提供了一种灵活的方式。

@Resource {#Resource}

@Resource是Java EE(Jakarta EE)提供的注解,用于注入依赖,它可以用在字段或setter方法上,主要通过名称匹配来注入依赖。

@Resource的工作原理 {#Resource的工作原理}

  • 名称匹配@Resource首先通过名称匹配来查找bean。如果名称匹配失败,则通过类型匹配。
  • 简单配置@Resource不支持复杂的注入配置,如@Qualifier

@Resource使用示例 {#Resource使用示例}

|---------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 | import javax.annotation.Resource; import javax.ejb.Stateless; @Stateless public class CarService { @Resource private Engine engine; // 其他方法 } public class Engine { // Engine的实现 } |

在这个示例中,CarService使用@Resource注解来注入Engine对象。

@Autowired@Resource的对比 {#Autowired和-Resource的对比}

提供者 {#提供者}

  • @Resource是Java EE规范的一部分,适用于任何兼容的Java EE容器。
    • @Autowired是Spring框架提供的注解,需要依赖 Spring框架,同时也支持@Resource

匹配策略 {#匹配策略}

  • @Autowired主要通过类型匹配,必要时通过@Qualifier进一步指定。
  • @Resource主要通过名称匹配,然后才是类型匹配。

配置复杂度 {#配置复杂度}

  • @Autowired支持更多的配置选项,如@Primary@Qualifier
  • @Resource配置相对简单,不支持复杂配置。

框架依赖 {#框架依赖}

  • @Autowired是Spring特有的注解,需要依赖Spring框架。
  • @Resource是Java EE规范的一部分,适用于任何兼容的Java EE容器。

使用场景 {#使用场景}

  • 如果使用 Spring框架,@Autowired是更常用的选择,因为它提供了更多功能。
  • 在Java EE环境中,@Resource是标准的选择。

总结 {#总结}

@Autowired@Resource是 Java开发中实现依赖注入的两种常用方式。虽然它们都可以用于自动装配 bean,但在匹配策略、配置复杂度和框架依赖上存在显著差异。因此,理解两者的原理和差异,可以帮助我们更好地理解 Java依赖注入的机制,更好的应用这两个注解。

交流学习 {#交流学习}

最后,把猿哥的座右铭送给你:投资自己才是最大的财富。 如果你觉得文章有帮助,请帮忙转发给更多的好友,或关注公众号:猿java,持续输出硬核文章。

赞(5)
未经允许不得转载:工具盒子 » @AutoWired和 @Resource原理分析!