为什么推荐使用构造函数注入而非@Autowired注解进行字段注入

server/2024/12/27 11:52:17/

在 Spring 框架中,推荐使用构造函数注入而非@Autowired注解进行字段注入,主要有以下几个原因:

1. 依赖不可变和空指针安全

  • 构造函数注入:使用构造函数注入时,依赖在对象创建时就必须提供,一旦对象创建完成,其依赖关系就固定下来,不会再改变。这有助于确保对象的状态在整个生命周期中都是一致和可预测的。而且,由于依赖是在构造函数中注入的,在使用对象之前,依赖已经被初始化,不会出现空指针异常。
    • 举例:假设有一个UserService类,它依赖于UserRepository。使用构造函数注入的方式如下:
public class UserService {private final UserRepository userRepository;public UserService(UserRepository userRepository) {this.userRepository = userRepository;}// 业务方法public User findUserById(Long id) {return userRepository.findById(id);}
}

在这个例子中,UserService在创建时就必须传入UserRepository实例,之后在调用findUserById方法时,userRepository肯定已经被初始化,不会出现空指针问题。

  • 字段注入:使用@Autowired注解进行字段注入时,依赖是在对象创建之后通过反射机制注入的。这就意味着在对象创建和依赖注入之间存在一个短暂的时间窗口,在这个窗口内,依赖可能为空。如果在依赖注入完成之前就访问依赖,就会导致空指针异常。
    • 举例:同样是UserService类,使用字段注入的方式如下:
public class UserService {@Autowiredprivate UserRepository userRepository;// 业务方法public User findUserById(Long id) {return userRepository.findById(id);}
}

在这个例子中,如果UserService在构造函数或者其他方法中过早地访问userRepository,而此时依赖注入还未完成,就可能会抛出空指针异常。

2. 便于单元测试

  • 构造函数注入:构造函数注入使得单元测试更加简单和直观。在进行单元测试时,只需要创建一个模拟的依赖对象,并通过构造函数传递给被测试对象即可。这使得测试代码与生产代码之间的依赖关系更加清晰,也更容易控制测试环境。
    • 举例:对于上述使用构造函数注入的UserService,单元测试可以这样写:
import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.*;
import static org.mockito.Mockito.*;public class UserServiceTest {@Testpublic void testFindUserById() {// 创建模拟的UserRepositoryUserRepository mockRepository = mock(UserRepository.class);when(mockRepository.findById(1L)).thenReturn(new User());// 通过构造函数创建UserService实例UserService userService = new UserService(mockRepository);// 调用方法进行测试User user = userService.findUserById(1L);assertNotNull(user);}
}
  • 字段注入:使用@Autowired注解进行字段注入时,在单元测试中需要使用反射或者其他复杂的技术来模拟依赖注入。这使得测试代码变得更加复杂,也增加了测试的难度和维护成本。
    • 举例:对于使用字段注入的UserService,单元测试可能需要这样写:
import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.*;
import static org.mockito.Mockito.*;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.context.ApplicationContext;
import org.springframework.test.context.ContextConfiguration;@SpringBootTest
@ContextConfiguration
public class UserServiceTest {@Autowiredprivate ApplicationContext applicationContext;@Testpublic void testFindUserById() {// 创建模拟的UserRepositoryUserRepository mockRepository = mock(UserRepository.class);when(mockRepository.findById(1L)).thenReturn(new User());// 通过反射设置UserService的userRepository字段UserService userService = applicationContext.getBean(UserService.class);try {java.lang.reflect.Field field = UserService.class.getDeclaredField("userRepository");field.setAccessible(true);field.set(userService, mockRepository);} catch (NoSuchFieldException | IllegalAccessException e) {e.printStackTrace();}// 调用方法进行测试User user = userService.findUserById(1L);assertNotNull(user);}
}

可以看到,字段注入的测试代码更加复杂,需要使用反射来设置依赖,增加了测试的难度和维护成本。

3. 更好的代码可读性和可维护性

  • 构造函数注入:构造函数注入将对象的依赖关系明确地展示在构造函数的参数列表中。这使得代码的阅读者能够一眼看出该对象依赖于哪些其他对象,以及这些依赖是如何被初始化的。这种清晰的依赖关系有助于提高代码的可读性和可维护性。
    • 举例:对于前面的UserService类,通过构造函数注入,我们可以清楚地看到UserService依赖于UserRepository,并且依赖是在构造函数中进行初始化的。
  • 字段注入:使用@Autowired注解进行字段注入时,依赖关系被隐藏在字段声明中,没有在构造函数或者其他明显的位置展示。这使得代码的阅读者需要花费更多的时间和精力来理解对象的依赖关系和初始化过程。
    • 举例:在使用字段注入的UserService类中,依赖关系通过@Autowired注解隐藏在字段声明中,不直观。


http://www.ppmy.cn/server/153607.html

相关文章

【MySQL】十四,MySQL 8.0的隐藏索引

在MySQL 8.0之前的版本中,索引只能直接删除。如果删除后发现引起了系统故障,又必须进行创建。当表的数据量比较大的时候,这样做的代价就会非常高。 在MySQL 8.0中,提供了隐藏索引。如果想删除某个索引,那么在实际删除…

上海AI中心记录

先谈谈做了什么项目,用了什么技术,有什么亮点和困难,采用什么方案解决的。 1、js事件循环 调用栈(Call Stack): JavaScript 是单线程的,所有的代码执行都是在调用栈中进行的。当函数被调用时&a…

SpringBoot的MVC接口增加签名

一、确定签名策略 HMAC(Hash-based Message Authentication Code):使用对称密钥。RSA:使用非对称密钥对(公钥/私钥)。OAuth:用于第三方授权和签名。 二、创建签名工具类 1、HMAC 签名工具类 …

Lambda、Stream流、线程(池)、IO

文章目录 LambdaStream流线程(池)IO Lambda 使用前提 必须存在一个接口接口中有且只有一个抽象方法 格式 : ( 形式参数 ) -> { 代码块 } 形式参数:如果有多个参数,参数之间用逗号隔开;如果没有参数,留空…

React Hooks

React Hooks React Hooks 是 React 16.8 版本中引入的一种新功能,它允许在不编写类的情况下使用状态和生命周期等特性。 在 Hooks 出现之前,React 里的函数式组件也被称为无状态组件。 函数组件和类组件的区别? 类组件必须要注意this指向…

08. 基于docker-compose部署LNMP架构

目录 前言 1、docker 1.1 任务要求 1.2 关闭防火墙 1.3 安装docker 1.4 配置镜像加速下载 2、Nginx 2.1 建立工作目录并进行相关操作 2.2 准备 nginx.conf 配置文件 3、Mysql 3.1 建立工作目录并进行相关操作 3.2 编写 my.cnf 配置文件 4、PHP 4.1 建立工作目录并…

使用Python实现量子计算应用:走进量子世界的大门

量子计算作为一种全新的计算范式,正在逐步改变我们的计算方式。与经典计算机依赖比特(bits)进行信息处理不同,量子计算机使用量子比特(qubits)进行计算,这使得量子计算在处理某些复杂问题上具有…

批量识别工作表中二维码信息-Excel易用宝

今天一大早,我们老板心急火燎的找到我,说是这个表格中的商品编码都不见了,问我能不能通过商品二维码还原商品编码,做好了中午给我加个鸡腿。 哎呀,这活不简单啊,我勉为其难的说,我先试试吧。 等…