Spring Boot单元测试实战指南:从零到高效测试

      在Spring Boot开发中,单元测试是保障代码质量的核心环节。本文将基于实际开发场景,手把手教你如何快速实现分层测试、模拟依赖、编写高效断言,并分享最佳实践!


一、5分钟环境搭建

  1. 添加依赖
    pom.xml中引入spring-boot-starter-test,默认集成JUnit 5、Mockito、AssertJ等工具:

    <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope>
    </dependency>
    
  2. 测试类模板
    测试类通常放在src/test/java目录,命名规范为被测试类名 + Test。基础模板如下:

    @SpringBootTest // 加载Spring上下文(适合集成测试)
    @ExtendWith(SpringExtension.class) // JUnit 5注解
    class UserServiceTest {@Autowiredprivate UserService userService;// 测试方法...
    }
    

二、分层测试:精准覆盖每一层

1. Service层测试:Mock依赖
场景:测试业务逻辑,隔离外部依赖(如数据库、第三方接口)。
工具:@MockBean模拟Spring管理的Bean。

@SpringBootTest
class UserServiceTest {@Autowiredprivate UserService userService;@MockBean // 模拟Repository依赖private UserRepository userRepository;@Testvoid testGetUserById() {// 1. 定义Mock行为when(userRepository.findById(1L)).thenReturn(Optional.of(new User("Alice")));// 2. 调用方法User user = userService.getUserById(1L);// 3. 断言结果assertThat(user.getName()).isEqualTo("Alice");}
}

2. Controller层测试:MockMvc模拟HTTP请求
工具:@WebMvcTest仅加载Web层组件,搭配MockMvc模拟请求。

@WebMvcTest(UserController.class) // 仅初始化Web层
class UserControllerTest {@Autowiredprivate MockMvc mockMvc;@MockBeanprivate UserService userService;@Testvoid testGetUser() throws Exception {when(userService.getUserById(1L)).thenReturn(new User("Alice"));// 模拟GET请求并验证响应mockMvc.perform(get("/users/1")).andExpect(status().isOk()).andExpect(jsonPath("$.name").value("Alice"));}
}

3. 持久层测试:真实数据库操作
工具:@DataJpaTest(JPA)或@DataMongoTest(MongoDB)。

@DataJpaTest // 仅加载JPA组件,默认使用H2内存数据库
class UserRepositoryTest {@Autowiredprivate UserRepository userRepository;@Testvoid testSaveUser() {User user = new User("Bob");User savedUser = userRepository.save(user);// 验证ID自动生成assertThat(savedUser.getId()).isNotNull();}
}

三、测试进阶技巧

1. 参数化测试:覆盖多场景
使用@ParameterizedTest快速测试多组输入:

@ParameterizedTest
@ValueSource(ints = {1, 3, 5, -1}) // 测试奇数和边界值
void testIsOdd(int number) {assertTrue(number % 2 != 0);
}

2. 测试生命周期控制
@BeforeEach:每个测试方法前执行(如初始化数据)。

@AfterEach:每个测试方法后执行(如清理数据库)。

@BeforeEach
void setup() {userRepository.deleteAll();userRepository.save(new User("TestUser"));
}

3. 测试专用配置
通过@TestPropertySource加载测试配置文件:

@SpringBootTest
@TestPropertySource(locations = "classpath:test.properties")
class MyTest {// 使用test.properties中的配置
}

四、最佳实践与避坑指南

  1. 分层策略
    • 单元测试:聚焦单一类,用@MockBean隔离外部依赖。

    • 集成测试:使用@SpringBootTest验证组件交互。

    • Web层测试:优先用@WebMvcTest而非启动完整应用。

  2. 断言工具选择
    • AssertJ(推荐):链式断言更直观。

    assertThat(userList).hasSize(3).extracting("name").contains("Alice", "Bob");
    
  3. 环境隔离
    • 使用H2内存数据库替代生产库。

    • 用@MockBean模拟外部服务(如支付接口)。

  4. 测试数据管理
    • @Sql注解:运行前执行SQL脚本初始化数据。

    @Test
    @Sql("/init-user-data.sql")
    void testUserCount() {assertThat(userRepository.count()).isEqualTo(5);
    }
    

五、快速运行测试

• IDE操作:右键测试类 → Run as JUnit Test

• 命令行运行:

mvn test         # Maven
./gradlew test   # Gradle

总结
      单元测试不是负担,而是高效开发的基石。通过分层测试、精准Mock和高效断言,可以大幅提升代码可靠性。立即动手,为你的Spring Boot项目加上“安全锁”吧!


新时代农民工

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/77053.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

React状态提升深度解析:原理、实战与最佳实践

一、状态提升的本质认知 React状态提升&#xff08;State Lifting&#xff09;是组件间通信的核心模式&#xff0c;其本质是通过组件树层级关系重构实现状态共享。与传统父子传参不同&#xff0c;它通过将状态提升到最近的共同祖先组件&#xff0c;建立单向数据流高速公路。 …

https nginx 负载均衡配置

我的系统是OpenEuler。 安装nginx yum install -y nginx 启动&开机启动 systemctl start nginx systemctl enable nginx 自定义conf配置文件 cat <<EOF >> /etc/nginx/conf.d/load_balancer.conf upstream backend {ip_hash; # 防止验证码验证失败server…

各种插值方法的Python实现

插值方法的Python实现 1. 线性插值&#xff08;Linear Interpolation&#xff09; 原理&#xff1a;用直线连接相邻数据点&#xff0c;计算中间点的值。 实现&#xff1a; import numpy as np from scipy.interpolate import interp1dx np.array([0, 1, 2, 3, 4]) y np.arr…

重新定义户外防护!基于DeepSeek的智能展开伞棚系统技术深度解析

从“手动操作”到“感知决策”&#xff0c;AI重构城市空间弹性 全球极端天气事件频发&#xff0c;传统伞棚依赖人工展开/收纳&#xff0c;存在响应滞后&#xff08;暴雨突袭时展开需3-5分钟&#xff09;、抗风能力弱&#xff08;8级风损毁率超60%&#xff09;、空间利用率低等痛…

Redis 基础和高级用法入门

redis 是什么&#xff1f; Redis是一个远程内存数据库&#xff0c;它不仅性能强劲&#xff0c;而且还具有复制特性以及为解决问题而生的独一无二的数据模型。Redis提供了5种不同类型的数据结构&#xff0c;各式各样的问题都可以很自然地映射到这些数据结构上&#xff1a…

常见数据库关键字示例 SQL 及执行顺序分析(带详细注释)

示例 SQL 及执行顺序分析&#xff08;带详细注释&#xff09; 示例 1&#xff1a;基础查询&#xff08;含多表关联、过滤、分组、排序&#xff09; SELECT -- 1. 选择字段&#xff08;包含聚合函数和别名&#xff09;e.department, COUNT(e.employee_id) AS total_employees, …

设计模式--建造者模式详解

建造者模式 建造者模式也属于创建型模式&#xff0c;它提供了一种创建对象的最佳方式 定义&#xff1a;将一个复杂对象的构建和它的表示分离&#xff0c;使得同样的构建过程可以创建不同的表示&#xff08;假设有不同的建造者实现类&#xff0c;可以产生不同的产品&#xff09…

PCB 过孔铜厚的深入指南

***前言&#xff1a;在上一期的文章中介绍了PCB制造的工艺流程&#xff0c;但仍然想在过孔的铜厚和PCB的过孔厚径比两个方面再深入介绍。 PCB铜厚的定义 电路中铜的厚度以盎司(oz)**表示。那么&#xff0c;为什么用重量单位来表示厚度呢? 盎司(oz)的定义 将1盎司(28.35 克)的铜…

如何配置 Conda 使用镜像源加速

如何配置 Conda 使用镜像源加速 为了提高使用 Anaconda 或 Miniconda 时包管理的速度&#xff0c;特别是在国内网络环境下&#xff0c;可以通过配置镜像源来实现更快的下载。以下是详细的步骤说明&#xff1a; 1. 安装 Conda&#xff08;如果尚未安装&#xff09; 如果你还没…

【k8s】k8s是怎么实现自动扩缩的

Kubernetes 提供了多种自动扩缩容机制&#xff0c;主要包括 Pod 水平自动扩缩&#xff08;HPA&#xff09;、垂直 Pod 自动扩缩&#xff08;VPA&#xff09; 和 集群自动扩缩&#xff08;Cluster Autoscaler&#xff09;。以下是它们的实现原理和配置方法&#xff1a; 1. Pod …

Reflex 完全指南:用 Python 构建现代 Web 应用的终极体验

“写 Python&#xff0c;就能构建 Web 前端。”——这不再是梦想&#xff0c;而是由 Reflex 带来的现实。 过去&#xff0c;构建一个现代 Web 应用意味着你要学会前端&#xff08;React/JS/HTML/CSS&#xff09; 后端&#xff08;Flask/Django&#xff09; API 交互&#xff08…

Vue实战(08)解决 Vue 项目中路径别名 `@` 在 IDE 中报错无法识别的问题

一、引言 ​ 在 Vue 项目开发过程中&#xff0c;路径别名是一个非常实用的特性&#xff0c;它能够帮助开发者简化文件引用路径&#xff0c;提高代码的可读性和可维护性。其中&#xff0c; 作为一个常见的路径别名&#xff0c;通常被用来指向项目的 src 目录。然而&#xff0c;…

5.学习笔记-SpringMVC(P61-P70)

SpringMVC-SSM整合-接口测试 (1)业务层接口使用junit接口做测试 (2)表现层用postman做接口测试 (3)事务处理— 1&#xff09;在SpringConfig.java&#xff0c;开启注解&#xff0c;是事务驱动 2&#xff09;配置事务管理器&#xff08;因为事务管理器是要配置数据源对象&…

[论文阅读]REPLUG: Retrieval-Augmented Black-Box Language Models

REPLUG: Retrieval-Augmented Black-Box Language Models REPLUG: Retrieval-Augmented Black-Box Language Models - ACL Anthology NAACL-HLT 2024 在这项工作中&#xff0c;我们介绍了RePlug&#xff08;Retrieve and Plug&#xff09;&#xff0c;这是一个新的检索增强型…

Mysql的深度分页查询优化

一、深度分页为什么慢&#xff1f; 当执行 SELECT * FROM orders ORDER BY id LIMIT 1000000, 10 时&#xff1a; MySQL 会扫描前 1,000,010 行&#xff0c;丢弃前 100 万行&#xff0c;仅返回 10 行。偏移量&#xff08;offset&#xff09;越大&#xff0c;扫描行数越多&…

最新扣子(Coze)案例教程:Excel数据生成统计图表,自动清洗数据+转换可视化图表+零代码,完全免费教程

大家好&#xff0c;我是斜杠君。 知识星球群有同学和我说每天的工作涉及很多数据表的重复操作&#xff0c;想学习Excel数据表通过大模型自动转数据图片的功能。 今天斜杠君就带大家一起搭建一个智能体&#xff0c;以一个销售行业数据为例&#xff0c;可以快速实现自动清洗Exc…

Uniapp 中缓存操作指南

在 Uniapp 中,你可以使用三种方式操作缓存:同步方法、异步方法和 Vuex 持久化存储。以下是详细的设置、获取和清除缓存的方法: 1. 同步方法 设置缓存 uni.setStorageSync(key, value); // 示例 uni.setStorageSync(token, abc123); 获取缓存 const value = uni.getStor…

k8s的yaml文件里的volume跟volumeMount的区别

volume 是 Pod 级别的资源&#xff0c;用于定义存储卷。它是一个独立于容器的存储资源&#xff0c;可以被一个或多个容器共享使用。volume 的定义位于 Pod 的 spec.volumes 部分。 特点 独立性&#xff1a;volume 是 Pod 的一部分&#xff0c;而不是容器的一部分。它独立于容…

梅毒单阳能否通过国企体检?

国企体检通常会参照公务员体检标准进行&#xff0c;梅毒检测是其中的常规项目。 一、明确“梅毒单阳”的定义 检测指标解析 TPPA阳性RPR阴性&#xff1a;可能为既往感染已治愈&#xff0c;或极早期/晚期梅毒&#xff1b; RPR阳性TPPA阴性&#xff1a;需警惕假阳性&#xff08…

Python 爬虫实战 | 企名科技

文章目录 一、企名科技1、目标网站2、网站特点3、确定解密位置4、扣js代码 一、企名科技 1、目标网站 网址&#xff1a;https://wx.qmpsee.com/articleDetail?idfeef62bfdac45a94b9cd89aed5c235be目标数据&#xff1a;获取消费行业研究下面的13篇文章数据 2、网站特点 服…