当前位置: 首页 > article >正文

SpringBoot单元测试实战:从Service到Controller的Mock技巧全解析

SpringBoot单元测试实战从Service到Controller的Mock技巧全解析单元测试是保障代码质量的重要防线但在实际开发中许多团队往往因为时间压力或技术复杂度而忽视这一环节。SpringBoot作为Java生态中最流行的框架之一其单元测试能力却常常被开发者低估。本文将带你深入SpringBoot单元测试的核心技巧从基础的Service层测试到复杂的Controller层Mock再到Repository层的巧妙处理通过真实案例演示如何构建高效、稳定的测试体系。1. 单元测试基础与环境搭建在开始具体层次的测试之前我们需要先理解SpringBoot单元测试的基本原理和必要准备。不同于传统的JUnit测试SpringBoot测试需要特殊的上下文支持和依赖管理。首先确保你的项目已经包含必要的测试依赖。在Maven项目中pom.xml应该包含以下依赖dependency groupIdorg.springframework.boot/groupId artifactIdspring-boot-starter-test/artifactId scopetest/scope /dependency这个starter包实际上是一个元依赖它包含了多个强大的测试工具JUnit 5现代Java测试框架的核心Spring TestSpring上下文支持MockitoMock/Spy对象创建AssertJ流畅的断言APIHamcrest匹配器库提示SpringBoot 2.4版本默认使用JUnit 5与JUnit 4有显著差异建议新项目直接采用JUnit 5测试类的基本结构如下SpringBootTest class BasicTestExample { Test void contextLoads() { // 测试Spring上下文是否成功加载 } }2. Service层测试Mock的艺术Service层是业务逻辑的核心也是单元测试的重点。在这一层我们需要特别关注如何隔离外部依赖特别是数据库访问和第三方服务调用。2.1 典型Service测试模式考虑一个用户积分服务UserPointService它依赖UserRepository和PointRepositoryService public class UserPointService { Autowired private UserRepository userRepository; Autowired private PointRepository pointRepository; public int calculateUserPoints(Long userId) { User user userRepository.findById(userId) .orElseThrow(() - new UserNotFoundException(userId)); return pointRepository.findByUser(user) .stream() .mapToInt(Point::getAmount) .sum(); } }对应的测试类应该这样设计SpringBootTest class UserPointServiceTest { Autowired private UserPointService userPointService; MockBean private UserRepository userRepository; MockBean private PointRepository pointRepository; Test void calculateUserPoints_shouldReturnSum() { // 准备测试数据 User testUser new User(1L, testUser); ListPoint testPoints Arrays.asList( new Point(testUser, 100), new Point(testUser, 200) ); // 配置Mock行为 when(userRepository.findById(1L)).thenReturn(Optional.of(testUser)); when(pointRepository.findByUser(testUser)).thenReturn(testPoints); // 执行测试 int result userPointService.calculateUserPoints(1L); // 验证结果 assertThat(result).isEqualTo(300); // 验证交互 verify(userRepository).findById(1L); verify(pointRepository).findByUser(testUser); } }2.2 高级Mock技巧Mockito提供了多种高级功能来处理复杂场景参数匹配器当你不关心具体参数值时when(userRepository.findById(anyLong())).thenReturn(Optional.of(testUser));异常测试验证异常情况下的行为Test void calculateUserPoints_shouldThrowWhenUserNotFound() { when(userRepository.findById(anyLong())).thenReturn(Optional.empty()); assertThatThrownBy(() - userPointService.calculateUserPoints(1L)) .isInstanceOf(UserNotFoundException.class) .hasMessageContaining(1); }连续调用模拟方法多次调用的不同返回when(mockService.getStatus()) .thenReturn(init) .thenReturn(processing) .thenReturn(done);3. Controller层测试MockMvc的威力Controller层的测试需要模拟HTTP请求和验证响应。SpringBoot提供了强大的MockMvc工具来实现这一点。3.1 基本Controller测试考虑一个简单的用户信息接口RestController RequestMapping(/api/users) public class UserController { Autowired private UserService userService; GetMapping(/{id}) public ResponseEntityUser getUser(PathVariable Long id) { return userService.findUserById(id) .map(ResponseEntity::ok) .orElse(ResponseEntity.notFound().build()); } }对应的测试类WebMvcTest(UserController.class) class UserControllerTest { Autowired private MockMvc mockMvc; MockBean private UserService userService; Test void getUser_shouldReturn200WhenFound() throws Exception { User testUser new User(1L, testUser); when(userService.findUserById(1L)).thenReturn(Optional.of(testUser)); mockMvc.perform(get(/api/users/1)) .andExpect(status().isOk()) .andExpect(jsonPath($.id).value(1L)) .andExpect(jsonPath($.name).value(testUser)); } Test void getUser_shouldReturn404WhenNotFound() throws Exception { when(userService.findUserById(anyLong())).thenReturn(Optional.empty()); mockMvc.perform(get(/api/users/999)) .andExpect(status().isNotFound()); } }3.2 复杂请求测试对于更复杂的请求如POST请求带JSON体Test void createUser_shouldReturn201() throws Exception { User newUser new User(null, newUser); User savedUser new User(1L, newUser); when(userService.createUser(any(User.class))).thenReturn(savedUser); mockMvc.perform(post(/api/users) .contentType(MediaType.APPLICATION_JSON) .content({\name\:\newUser\})) .andExpect(status().isCreated()) .andExpect(header().string(Location, /api/users/1)); }4. Repository层测试嵌入式数据库方案Repository层的测试通常需要真实的数据库交互。SpringBoot提供了内存数据库支持可以完美解决这个问题。4.1 基本Repository测试DataJpaTest class UserRepositoryTest { Autowired private TestEntityManager entityManager; Autowired private UserRepository userRepository; Test void findByName_shouldReturnMatchingUsers() { // 准备测试数据 User user1 new User(null, Alice); User user2 new User(null, Bob); User user3 new User(null, Alice); entityManager.persist(user1); entityManager.persist(user2); entityManager.persist(user3); entityManager.flush(); // 执行查询 ListUser alices userRepository.findByName(Alice); // 验证结果 assertThat(alices).hasSize(2); assertThat(alices).extracting(User::getName) .containsOnly(Alice); } }4.2 自定义查询测试对于自定义的JPQL或原生SQL查询public interface UserRepository extends JpaRepositoryUser, Long { Query(SELECT u FROM User u WHERE u.name LIKE %:keyword%) ListUser searchByName(Param(keyword) String keyword); } Test void searchByName_shouldReturnPartialMatches() { User user1 new User(null, Alice); User user2 new User(null, Bob); User user3 new User(null, Alicia); entityManager.persist(user1); entityManager.persist(user2); entityManager.persist(user3); ListUser results userRepository.searchByName(lic); assertThat(results).hasSize(2); assertThat(results).extracting(User::getName) .containsExactlyInAnyOrder(Alice, Alicia); }5. Mock与Spy的进阶应用在实际项目中我们经常需要更精细的控制测试行为。Mockito的Spy功能可以部分模拟真实对象保留部分真实行为。5.1 Spy基础用法SpringBootTest class OrderServiceTest { SpyBean private EmailService emailService; Autowired private OrderService orderService; Test void placeOrder_shouldSendEmail() { Order order new Order(testexample.com, Item1); // 让真实方法执行但记录调用 doCallRealMethod().when(emailService).sendConfirmation(anyString()); orderService.placeOrder(order); verify(emailService).sendConfirmation(testexample.com); } }5.2 复杂Spy场景有时我们需要修改Spy对象的部分行为Test void processBatch_shouldLogErrors() { ListItem items Arrays.asList( new Item(valid1), new Item(invalid), new Item(valid2) ); // 让真实方法处理大多数情况 doCallRealMethod().when(itemProcessor).process(any(Item.class)); // 但对特定输入抛出异常 doThrow(new ProcessingException(invalid item)) .when(itemProcessor).process(argThat(item - invalid.equals(item.getName()))); BatchResult result batchService.processBatch(items); assertThat(result.getSuccessCount()).isEqualTo(2); assertThat(result.getErrorCount()).isEqualTo(1); // 验证错误日志 verify(logger).error(Processing failed for item: invalid); }6. 测试最佳实践与常见陷阱在实际项目中应用这些技术时有几个关键点需要注意测试隔离每个测试应该独立运行不依赖其他测试的状态。使用BeforeEach初始化测试数据而不是依赖共享状态。测试命名采用一致的命名约定如methodName_scenario_expectedResult模式提高测试可读性。避免过度MockMock应该用于外部依赖而不是系统内部组件。过度Mock会导致测试与实现细节耦合。性能考虑SpringBootTest会加载完整上下文可能很慢。对于纯单元测试考虑使用更轻量级的ExtendWith(MockitoExtension.class)。常见陷阱忘记验证Mock交互过度指定Mock行为导致脆弱测试忽略异常场景测试测试代码重复度高// 不好的例子过度指定Mock行为 when(userRepository.findById(1L)) .thenReturn(Optional.of(new User(1L, Alice, aliceexample.com))); // 更好的方式只关注测试需要的属性 when(userRepository.findById(1L)) .thenReturn(Optional.of(new User(1L, Alice)));在实际项目中我发现最有效的测试策略是先为关键业务逻辑编写测试逐步扩展到边缘场景定期重构测试代码保持其可维护性将测试作为设计工具而不仅仅是验证工具

相关文章:

SpringBoot单元测试实战:从Service到Controller的Mock技巧全解析

SpringBoot单元测试实战:从Service到Controller的Mock技巧全解析 单元测试是保障代码质量的重要防线,但在实际开发中,许多团队往往因为时间压力或技术复杂度而忽视这一环节。SpringBoot作为Java生态中最流行的框架之一,其单元测试…...

Golang如何把日志写到文件_Golang日志文件教程【秒懂】

Go log包默认只输出到stderr,需用os.OpenFile创建*os.File(实现io.Writer)传给log.SetOutput;并发写安全但格式易乱;需手动flush或用bufio.NewWriter;长期运行需日志轮转等高级功能。Go 标准库的 log 包默认…...

如何排除AWR中的Idle Events_过滤不需要关注的网络等待与定时器

Idle Events是Oracle中wait_class为Idle的等待事件,如SQL*Net message from client等,它们反映客户端空闲状态而非数据库性能问题,若不剔除会干扰AWR分析、误导DBA定位假热点。什么是 Idle Events,为什么它们会干扰 AWR 分析awr 报…...

在Rockchip RK3588开发板上,用MPP库把H264视频转成YUV文件(附完整代码和避坑点)

在RK3588开发板上用MPP实现H264到YUV的高效解码实战 当我在RK3588开发板上第一次成功将H264视频流实时解码为YUV420SP格式时,那种成就感至今难忘。作为瑞芯微旗舰级芯片,RK3588的媒体处理能力在嵌入式领域堪称强悍,而MPP(Media Pr…...

自定义常用代码编辑快捷键大全

自定义适用、常用代码编辑快捷键通用编程快捷键代码提示快捷键文件操作快捷键编译器快捷键终端操作快捷键通用编程快捷键 格式化代码:Ctrl K注释/取消注释:Ctrl / , 添加注释:Ctrl shift /删除行代码:Ctrl D &…...

PyTorch 2.9 镜像部署全攻略:Jupyter和SSH两种方式任你选

PyTorch 2.9 镜像部署全攻略:Jupyter和SSH两种方式任你选 1. PyTorch 2.9 镜像概述 PyTorch 2.9 是一个开源的 Python 机器学习库,基于 Torch 库开发,底层由 C 实现,广泛应用于人工智能领域,特别是计算机视觉和自然语…...

测试用例模版与标准规范

一、测试用例通用模版(支持自动化生成与质量判定)字段名必填说明与格式要求自动化生成来源质量判定规则用例ID是{层级}_{类型}_{模块}_{序号}例:SYS_IT_ORDER_001自动生成唯一性、格式正则校验用例标题是[类型][正向/异常] 操作预期效果例&am…...

OpenTiny NEXT 从入门到精通·第 1 篇

OpenTiny NEXT 从入门到精通第 1 篇:启程篇——初识 OpenTiny NEXT,开启企业级智能前端之旅前端技术日新月异,Vue 2 要停止维护了,团队还在 Vue 2/3 多版本并存中挣扎;业务方希望接入 AI,前端却不知道怎么开…...

S2-Pro模型安全与合规应用指南:内容过滤与偏见缓解策略

S2-Pro模型安全与合规应用指南:内容过滤与偏见缓解策略 1. 企业级AI部署的安全挑战 当企业考虑部署S2-Pro这类大语言模型时,安全与合规问题往往成为首要考量。不同于个人使用场景,企业应用需要面对更严格的监管要求、更复杂的用户群体以及更…...

OpenTiny NEXT 从入门到精通·第 2 篇

OpenTiny NEXT 从入门到精通第 2 篇:组件篇——TinyVue 核心组件库深度实战组件库是前端应用的“乐高积木”。TinyVue 作为 OpenTiny 生态的核心 UI 组件库,拥有 130 企业级组件,覆盖中后台开发的绝大部分场景。但会用组件只是第一步&#xf…...

OpenClaw敏感数据处理:Qwen3-32B私有镜像的加密通信方案

OpenClaw敏感数据处理:Qwen3-32B私有镜像的加密通信方案 1. 为什么需要加密通信方案 上周帮朋友部署OpenClaw自动化财务对账流程时,遇到一个棘手问题:当系统自动处理银行流水和发票PDF时,这些敏感数据在传输过程中竟然以明文形式…...

计算机中级-数据库系统工程师-操作系统-进程管理(2)

一、死锁1. 死锁的定义现象描述: 指两个以上的进程互相都要求对方已经占有的资源,导致无法继续运行下去的现象。核心特征: 多个进程互相等待对方释放已获得的资源,导致所有进程都在无限等待。1)例题:死锁示例案例说明: 5个进程(A-E)每个需要4…...

QTQK-FJYJNDL-V137 远动及光伏群调群控装置技术介绍

一、产品概述QTQK-FJYJNDL-V137 群调群控装置由福建亿捷能电力科技公司自主研发,主要应用于光伏电站、风电场、储能电站、配电站等各类分布式电源场景,是一款专用的边缘计算与调控设备。该装置可实现现场测控装置、保护装置、逆变器、储能变流器等设备的…...

IBGP全互联实验+路由反射实验

知识点回顾IBGP全互联路由反射器实验先做IBGP全互联实验查看OSPF邻居关系R1上查看BGP邻居在R2上查看BGP邻居在R3上查看BGP邻居在R5上查看BGP邻居在R4上查看是否有去往10.10.5.5的路由在R5上查看是否有去往10.10.4.4的路由现在网络就能通了再做路由器反射实验在R1上跟R3上删除BG…...

基于Ubuntu20.04的SenseVoice-Small高性能部署方案

基于Ubuntu20.04的SenseVoice-Small高性能部署方案 语音识别技术正逐渐成为人机交互的重要桥梁,而如何在生产环境中高效部署模型成为很多开发者的实际需求。本文将手把手带你完成SenseVoice-Small在Ubuntu20.04系统上的高性能部署。 1. 环境准备与系统优化 在开始部…...

SpringBoot的生命周期原理分析之一SpringBoot准备容器与环境

目录 1.SpringBootApplication准备 1.1SpringApplication创建 1.2.1保存主配置源 1.2.2推断Web环境 1.2.3设置初始化器 1.2.4设置监听器 1.2.5确定主启动类 1.2.6扩展了解:SpringBoot的发展 1.2SpringBootApplication启动 1.2.1启动计时与全局异常处理机制…...

Wan2.2-I2V-A14B实操手册:WebUI中ControlNet风格控制与运动强度调节

Wan2.2-I2V-A14B实操手册:WebUI中ControlNet风格控制与运动强度调节 1. 环境准备与快速部署 在开始探索Wan2.2-I2V-A14B的强大功能前,我们需要确保环境已正确部署。本镜像已针对RTX 4090D 24GB显卡进行深度优化,开箱即用。 1.1 一键启动We…...

OpenClaw深度集成:千问3.5-9B作为默认推理引擎

OpenClaw深度集成:千问3.5-9B作为默认推理引擎 1. 为什么选择千问3.5-9B作为默认模型 去年冬天第一次接触OpenClaw时,我花了两周时间反复测试不同模型的适配性。当时用OpenAI的接口虽然方便,但每次截图识别、文件操作都要消耗大量token&…...

港科资讯|香港科大内地办(北京)赴宜参与第五届“330“三峡人才日活动 共探协同创新新路径

2026年3 月29-30日,香港科大内地办(北京)袁冶主任一行受邀参加[第五届宜昌“330”三峡人才日]系列活动,深度对接宜昌人才生态、产业布局与创新资源,共探协同创新新路径。“330” 三峡人才日源自葛洲坝水利枢纽工程 “330 工程” 历史符号&…...

【nginx】深入解析net::ERR_CONTENT_LENGTH_MISMATCH 200:权限配置与日志排查实战

1. 错误现象与初步诊断 当你用浏览器访问Nginx托管的网站时,突然看到控制台报错net::ERR_CONTENT_LENGTH_MISMATCH 200,但页面居然还能正常显示部分内容,这种情况是不是很诡异?我第一次遇到时也是一头雾水。这个错误表面看是内容长…...

从游戏背包到物流集装箱:深入浅出图解三维装箱问题(3D-BPP)

从游戏背包到物流集装箱:深入浅出图解三维装箱问题(3D-BPP) 想象一下你在玩《我的世界》,背包里塞满了钻石镐、金苹果和各种矿石,突然发现空间不够了——这时候你下意识做的事情,和亚马逊仓库的机器人分拣货…...

排序(五)【数据结构】

快速排序 核心思想 将待排序序列,围绕着基本值分成两部分,左边部分都小于基准值,右边部分都大于基准值 第一种方法:递归 优点:简单 缺点:需要单独开辟辅助空间brr数组 第二种方法:挖空法(很重要&…...

Davinci NvM Block与Fee Block关联配置详解

1. Davinci配置工具中的NvM与Fee Block基础概念 第一次接触Davinci配置工具时,很多人会对NvM Block和Fee Block的关系感到困惑。简单来说,NvM(Non-volatile Memory)Block是我们配置的非易失性存储单元,而Fee&#xff0…...

如何快速上手AssetStudio:Unity游戏资源提取的终极指南

如何快速上手AssetStudio:Unity游戏资源提取的终极指南 【免费下载链接】AssetStudio AssetStudio - Based on the archived Perfares AssetStudio, I continue Perfares work to keep AssetStudio up-to-date, with support for new Unity versions and additional…...

程序员效率工具:Yi-Coder-1.5B部署与真实任务测试报告

程序员效率工具:Yi-Coder-1.5B部署与真实任务测试报告 还在为写一个简单的文件处理脚本而翻遍搜索引擎吗?或者面对一段陌生的遗留代码,需要花半小时去理解它的逻辑?对于程序员来说,日常开发中充斥着大量重复、琐碎但必…...

避坑指南:用C++在ROS2中实现LOAM建图与定位时,如何解决PCL、Eigen和g2o的版本兼容与编译问题

ROS2环境下LOAM算法实战:PCL、Eigen与g2o版本兼容性深度解决方案 当你在ROS2环境中实现LOAM(Lidar Odometry and Mapping)算法时,PCL、Eigen和g2o这三个关键库的版本兼容性问题往往会成为项目推进的最大障碍。本文将深入剖析这些依…...

22 华夏之光永存:指挥AI修复自身代码bug,无需人工逐行查找

指挥AI修复自身代码bug,无需人工逐行查找 摘要 本文为《30天掌控AI编程:从指令到落地,手把手教你指挥AI写代码》系列第二十二篇,属于第四阶段「AI代码校验与优化」核心内容。承接上篇AI代码校验成果,本篇聚焦AI代码bug自动化修复,针对零基础开发者“不会改bug、改完又出…...

OpenClaw异常处理设计:Qwen3.5-9B图片任务失败自动恢复方案

OpenClaw异常处理设计:Qwen3.5-9B图片任务失败自动恢复方案 1. 为什么需要异常处理机制? 上周我尝试用OpenClawQwen3.5-9B实现证件照自动裁剪时,遇到了典型的"三连击"问题:网络波动导致图片上传中断、模型响应超时、输…...

seo推广员如何进行用户体验优化_seo推广员的工作内容有哪些

SEO推广员如何进行用户体验优化 在当今的数字化时代,用户体验(UX)已经成为网站运营和SEO推广的重要组成部分。一个优秀的用户体验不仅能够提高用户的满意度和忠诚度,还能直接影响网站的SEO表现。作为一名SEO推广员,如…...

Qwen3-14B镜像快速入门:内置模型+完整环境,开箱即用教程

Qwen3-14B镜像快速入门:内置模型完整环境,开箱即用教程 1. 为什么选择Qwen3-14B镜像 在AI模型部署过程中,环境配置往往是最耗时的环节。传统部署方式需要手动安装CUDA、PyTorch、模型权重等数十个组件,版本兼容性问题频发&#…...