Java 新手入门:依赖注入的 N 种姿势,总有一款适合你!

简介: Java 新手入门:依赖注入的 N 种姿势,总有一款适合你!

在 Spring 的世界里,依赖注入就像一座桥梁,将不同的组件连接起来,构建出一个松耦合、易维护的应用程序。今天我们就来聊聊 Spring 依赖注入的那些事儿,带你轻松掌握这项必备技能!

什么是依赖注入?

简单来说,依赖注入就是将一个对象所依赖的其他对象,通过外部的方式传递进来,而不是由对象本身来创建。这样做的好处是可以降低代码耦合度,提高代码的可重用性和可测试性。


假设你正在开发一个电商网站,其中有一个订单服务 OrderService,它依赖于商品服务 ProductService 和用户服务 UserService 来完成下单操作。

1. 传统方式 (不使用依赖注入):

public class OrderService {
 
    private ProductService productService = new ProductService(); // 自己创建依赖
    private UserService userService = new UserService();
 
    public void placeOrder(Order order) {
        // ... 获取商品信息
        Product product = productService.getProductById(order.getProductId());
        // ... 获取用户信息
        User user = userService.getUserById(order.getUserId());
        // ... 创建订单
        // ...
    }
}

这种方式的缺点很明显:

  • OrderService 强依赖于 ProductService 和 UserService 的具体实现,如果要替换成其他实现,就需要修改 OrderService 的代码。
  • 代码难以测试,因为你无法轻易地替换 ProductService 和 UserService 的模拟实现进行单元测试。

2. 依赖注入方式:

public class OrderService {
 
    @Autowired // 声明依赖,由 Spring 容器注入
    private ProductService productService;
 
    @Autowired
    private UserService userService;
 
    public void placeOrder(Order order) {
        // ... 获取商品信息
        Product product = productService.getProductById(order.getProductId());
        // ... 获取用户信息
        User user = userService.getUserById(order.getUserId());
        // ... 创建订单
        // ...
    }
}

使用依赖注入后:

  • OrderService 只依赖于 ProductService 和 UserService 的接口,而不依赖于它们的具体实现,降低了代码耦合度
  • 你可以通过配置文件或注解轻松地替换 ProductService 和 UserService 的实现,提高了代码的灵活性和可配置性。
  • 你可以方便地注入 ProductService 和 UserService 的模拟实现进行单元测试,提高了代码的可测试性。

Spring 主要提供了以下三种依赖注入的方式:

1. 构造器注入:

通过构造方法将依赖的对象传递给目标对象。

// 定义 CoffeeBean 类
public class CoffeeBean {
    // ... 咖啡豆相关属性和方法
}
 
// 定义 Water 类
public class Water {
    // ... 水相关属性和方法
}
 
// 定义 CoffeeMachine 类
public class CoffeeMachine {
    // ... 咖啡机相关属性和方法
}
 
// 定义 CoffeeMaker 类
public class CoffeeMaker {
 
    private final CoffeeBean coffeeBean; // 声明依赖,不可变
    private final Water water;
    private final CoffeeMachine coffeeMachine;
 
    // 通过构造方法注入依赖
    public CoffeeMaker(CoffeeBean coffeeBean, Water water, CoffeeMachine coffeeMachine) {
        this.coffeeBean = coffeeBean;
        this.water = water;
        this.coffeeMachine = coffeeMachine;
    }
 
    // ... 其他方法,例如 makeCoffee()
}

代码解读:

  • CoffeeMaker 类依赖于 CoffeeBean、Water 和 CoffeeMachine 三个类。
  • 通过在 CoffeeMaker 的构造方法中声明这三个类的参数,并在创建 CoffeeMaker 对象时将它们传递进去,就实现了构造器注入。
  • 使用 final 关键字修饰依赖对象,可以确保它们在对象创建后不可修改,保证了依赖的完整性。

2. Setter 注入:

通过 Setter 方法将依赖的对象传递给目标对象。

public class CoffeeMaker {
 
    private CoffeeBean coffeeBean;
    private Water water;
    private CoffeeMachine coffeeMachine;
 
    // 通过 Setter 方法注入依赖
    @Autowired
    public void setCoffeeBean(CoffeeBean coffeeBean) {
        this.coffeeBean = coffeeBean;
    }
 
    @Autowired
    public void setWater(Water water) {
        this.water = water;
    }
 
    @Autowired
    public void setCoffeeMachine(CoffeeMachine coffeeMachine) {
        this.coffeeMachine = coffeeMachine;
    }
 
    // ... 其他方法
}

代码解读:

  • CoffeeMaker 类中定义了三个 Setter 方法,分别用于设置 coffeeBean、water 和 coffeeMachine 属性。
  • 在每个 Setter 方法上使用 @Autowired 注解,告诉 Spring 容器在创建 CoffeeMaker 对象后,自动调用这些 Setter 方法注入依赖的对象。

3. 字段注入:

直接在字段上使用  @Autowired  注解,让 Spring 自动注入依赖的对象。

public class CoffeeMaker {
 
    @Autowired
    private CoffeeBean coffeeBean;
 
    @Autowired
    private Water water;
 
    @Autowired
    private CoffeeMachine coffeeMachine;
 
    // ... 其他方法
}

代码解读:

  • 在 CoffeeMaker 类的字段上使用 @Autowired 注解,告诉 Spring 容器在创建 CoffeeMaker 对象后,自动为这些字段注入依赖的对象。
  • 这种方式最为简洁,但可读性稍差,而且无法在字段上使用 final 关键字。

三种方式的比较:

  • 构造器注入:
  • 当依赖的对象较多时,构造方法会变得臃肿。
  • 如果依赖关系发生变化,就需要修改构造方法。
  • 能够保证依赖的对象在对象创建时就已设置,避免出现 NullPointerException。
  • 代码更易于维护,因为依赖关系清晰明确。
  • Setter 注入:
  • 无法保证依赖的对象在对象创建时就已设置,可能导致 NullPointerException。
  • 代码可读性稍差,因为依赖关系分散在各个 Setter 方法中。
  • 灵活,可以根据需要选择性地注入依赖的对象。
  • 如果依赖关系发生变化,只需要添加或修改对应的 Setter 方法即可。
  • 字段注入:
  • 可读性较差,容易与其他代码混淆。
  • 无法在字段上使用 final 关键字,无法保证依赖的完整性。
  • 简洁,代码量少。

总结

依赖注入是 Spring 框架的核心机制之一,它可以帮助我们编写更加松耦合、易测试、易维护的代码。三种依赖注入方式各有优缺点,选择哪种方式取决于具体的需求和编码习惯。


希望这篇公众号文章能够帮助你更好地理解 Spring 依赖注入的不同方式,并在实际开发中灵活运用

相关文章
|
9天前
|
Java 程序员 UED
Java中的异常处理:从入门到精通
【9月更文挑战第23天】在Java编程的世界中,异常是程序执行过程中不可避免的事件,它们可能会中断正常的流程并导致程序崩溃。本文将通过浅显易懂的方式,引导你理解Java异常处理的基本概念和高级技巧,帮助你编写更健壮、更可靠的代码。我们将一起探索如何捕获和处理异常,以及如何使用自定义异常来增强程序的逻辑和用户体验。无论你是初学者还是有一定经验的开发者,这篇文章都将为你提供有价值的见解和实用的技巧。
28 4
|
2月前
|
设计模式 前端开发 Java
【前端学java】SpringBootWeb极速入门-分层解耦(03)
【8月更文挑战第13天】SpringBootWeb极速入门-分层解耦(03)
17 2
【前端学java】SpringBootWeb极速入门-分层解耦(03)
|
2月前
|
开发框架 前端开发 Java
【前端学java】SpringBootWeb极速入门-实现一个简单的web页面01
【8月更文挑战第12天】SpringBootWeb极速入门-实现一个简单的web页面01
53 3
【前端学java】SpringBootWeb极速入门-实现一个简单的web页面01
|
2月前
|
JSON 前端开发 Java
【前端学java】SpringBootWeb极速入门-请求参数解析(02)
【8月更文挑战第12天】SpringBootWeb极速入门-请求参数解析(02)
17 1
【前端学java】SpringBootWeb极速入门-请求参数解析(02)
|
2月前
|
算法 Java 开发者
Java 编程入门:从零到一的旅程
本文将带领读者开启Java编程之旅,从最基础的语法入手,逐步深入到面向对象的核心概念。通过实例代码演示,我们将一起探索如何定义类和对象、实现继承与多态,并解决常见的编程挑战。无论你是编程新手还是希望巩固基础的开发者,这篇文章都将为你提供有价值的指导和灵感。
|
2月前
|
存储 Java 程序员
Java中的集合框架:从入门到精通
【8月更文挑战第30天】在Java的世界里,集合框架是一块基石,它不仅承载着数据的存储和操作,还体现了面向对象编程的精髓。本篇文章将带你遨游Java集合框架的海洋,从基础概念到高级应用,一步步揭示它的奥秘。你将学会如何选择合适的集合类型,掌握集合的遍历技巧,以及理解集合框架背后的设计哲学。让我们一起探索这个强大工具,解锁数据结构的新视角。
|
2月前
|
Java 程序员 UED
Java中的异常处理:从入门到精通
【8月更文挑战第28天】在Java编程的世界里,异常处理是一块基石,它确保了程序的健壮性和可靠性。本文将通过深入浅出的方式,带你了解Java异常处理的基本概念、分类、以及如何有效地捕获和处理异常。我们将一起探索try-catch-finally结构的奥秘,并学习如何使用throws关键字声明方法可能会抛出的异常。此外,我们还会讨论自定义异常类的创建和使用,以及最佳实践。无论你是Java新手还是有一定经验的开发者,这篇文章都将为你提供宝贵的知识,帮助你编写出更加稳定和可靠的代码。
|
2月前
|
编解码 网络协议 Oracle
java网络编程入门以及项目实战
这篇文章是Java网络编程的入门教程,涵盖了网络编程的基础知识、IP地址、端口、通讯协议(TCP和UDP)的概念与区别,并提供了基于TCP和UDP的网络编程实例,包括远程聊天和文件传输程序的代码实现。
java网络编程入门以及项目实战
|
22天前
|
Java 程序员
Java中的异常处理:从入门到精通
在Java编程的世界中,异常处理是保持程序稳定性和可靠性的关键。本文将通过一个独特的视角—把异常处理比作一场“捉迷藏”游戏—来探讨如何在Java中有效管理异常。我们将一起学习如何识别、捕捉以及处理可能出现的异常,确保你的程序即使在面对不可预见的错误时也能优雅地运行。准备好了吗?让我们开始这场寻找并解决Java异常的冒险吧!
|
2月前
|
前端开发 IDE Java
"揭秘前端转Java的秘径:SpringBoot Web极速入门,掌握分层解耦艺术,让你的后端代码飞起来,你敢来挑战吗?"
【8月更文挑战第19天】面向前端开发者介绍Spring Boot后端开发,通过简化Spring应用搭建,快速实现Web应用。本文以创建“Hello World”应用为例,展示项目基本结构与运行方式。进而深入探讨三层架构(Controller、Service、DAO)下的分层解耦概念,通过员工信息管理示例,演示各层如何协作及依赖注入的使用,以此提升代码灵活性与可维护性。
39 2
下一篇
无影云桌面