0

0

Mockito单元测试:正确模拟Optional返回值以避免业务异常

碧海醫心

碧海醫心

发布时间:2025-07-17 16:16:01

|

255人浏览过

|

来源于php中文网

原创

Mockito单元测试:正确模拟Optional返回值以避免业务异常

本文探讨了在使用Mockito对Java服务层方法进行单元测试时,因未正确模拟Optional类型返回值而导致业务异常的问题。通过分析Mockito的默认行为,本文详细解释了为何findById等方法会返回空Optional,并提供了明确的解决方案:通过when().thenReturn()显式地stub Optional返回值,确保测试流程按预期执行,从而避免“用户未找到”等业务异常,提升测试的准确性和可靠性。

问题分析:Mockito与Optional的默认行为

在进行单元测试时,我们通常会使用mocking框架(如mockito)来隔离被测试单元(system under test, sut)与外部依赖,确保测试的焦点仅在于sut本身的逻辑。然而,当依赖的方法返回optional类型时,如果不进行适当的模拟,可能会遇到意料之外的行为。

考虑以下服务层updateUser方法:

@Override
public UserDTO updateUser(String id, UserDTO updatedUser) {
    // 1. 根据updatedUser中的userName查找现有用户
    Optional databaseUser = userRepository.findById(Integer.valueOf(updatedUser.getUserName()));
    // 2. 如果用户不存在,抛出异常
    if (databaseUser.isEmpty()) {
        throw new UserNotFoundException("User with the this id is not found");
    }
    // 3. 映射DTO到实体
    UserEntity entity = mapToUserEntity(updatedUser);
    // 4. 保存更新后的实体
    return map(userRepository.save(entity));
}

该方法首先通过userRepository.findById()查询用户是否存在,如果不存在则抛出UserNotFoundException。

在编写updateUser方法的单元测试时,我们可能会遇到“User with the this id is not found”的异常,即使我们预期用户是存在的。这是因为Mockito在模拟对象时,对于返回复杂类型(如Optional、集合、自定义对象等)的方法,如果没有明确指定其行为,它会返回其类型的默认值。对于Optional,其默认值是Optional.empty()。

原始测试代码片段如下:

@Test
void updateUserTest(){
    // ... 用户DTO数据准备 ...

    // 模拟roleRepository.findById的行为
    when(roleRepository.findById(any())).thenReturn(Optional.of(new UserDTO().setId(roleId)));

    // 将userDto映射为UserEntity
    UserEntity userEntity = userService.mapToUserEntity(userDto);

    // 模拟userRepository.save的行为
    when(userRepository.save(any())).thenReturn(userEntity.setId(id));

    // 调用被测方法
    userService.updateUser(String.valueOf(id), userDto);
    var actualUser = userService.updateUser(String.valueOf(id), userDto); // 再次调用

    // ... 断言 ...
}

在这个测试中,userRepository.findById()方法并没有被显式地模拟(stub)。因此,当updateUser方法内部调用userRepository.findById(Integer.valueOf(updatedUser.getUserName()))时,Mockito会返回Optional.empty()。这导致databaseUser.isEmpty()判断为真,进而触发UserNotFoundException。

解决方案:显式Stubbing Optional返回值

要解决这个问题,核心在于明确告诉Mockito,当userRepository.findById()被调用时,它应该返回一个包含预期UserEntity的Optional对象,而不是默认的空Optional。

Bika.ai
Bika.ai

打造您的AI智能体员工团队

下载

我们需要在调用userService.updateUser之前,添加对userRepository.findById的模拟。模拟时需要注意findById的参数类型,它接收一个Integer类型的ID。在我们的userDto中,userName被设置为"12",因此Integer.valueOf(updatedUser.getUserName())会是12。

以下是正确的模拟方式:

// 1. 准备一个预期由findById返回的UserEntity实例
// 这个实例应该代表数据库中已存在的用户
UserEntity existingUserEntity = new UserEntity(); // 实际项目中应根据需要填充数据
existingUserEntity.setId(Integer.valueOf(userDto.getUserName())); // 确保ID与查询匹配
// ... 填充existingUserEntity的其他属性 ...

// 2. 模拟userRepository.findById的行为,使其返回包含existingUserEntity的Optional
when(userRepository.findById(Integer.valueOf(userDto.getUserName())))
    .thenReturn(Optional.of(existingUserEntity));

// 3. 模拟userRepository.save的行为(如果尚未模拟)
// 注意:userEntity是userService.mapToUserEntity(userDto)的结果,是将被保存的实体
UserEntity mappedUserEntity = userService.mapToUserEntity(userDto);
when(userRepository.save(any(UserEntity.class))).thenReturn(mappedUserEntity);

通过以上修改,当updateUser方法内部调用userRepository.findById(12)时,它将收到一个非空的Optional,从而跳过isEmpty()检查并继续执行后续的更新逻辑。

完整测试代码示例

为了提供一个更清晰、完整的测试示例,我们将整合上述修改,并对原始测试进行一些优化,例如避免重复调用被测方法。

import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.mockito.InjectMocks;
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;

import java.time.LocalDateTime;
import java.util.List;
import java.util.Optional;

import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.when;
import static org.mockito.Mockito.verify;

// 假设这些是你的Service, Repository和DTO类
// import com.example.service.UserService;
// import com.example.repository.UserRepository;
// import com.example.repository.RoleRepository;
// import com.example.dto.UserDTO;
// import com.example.entity.UserEntity;
// import com.example.exception.UserNotFoundException;

class UserServiceTest {

    @Mock
    private UserRepository userRepository;

    @Mock
    private RoleRepository roleRepository;

    @InjectMocks
    private UserService userService; // 假设UserService中包含mapToUserEntity和map方法

    @BeforeEach
    void setUp() {
        MockitoAnnotations.openMocks(this);
        // 如果UserService依赖于其他Service或组件,也需要在这里Mock并注入
        // 例如:userService = new UserService(userRepository, roleRepository, ...);
        // 或者使用 @Spy 来部分Mock真实对象
    }

    @Test
    void updateUser_shouldUpdateExistingUserSuccessfully() {
        // 1. 准备测试数据
        final int userId = 12; // 对应updatedUser.getUserName()
        final long roleId = 2L;

        UserDTO userDto = new UserDTO();
        userDto.setUserName(String.valueOf(userId)); // 用于findById的ID
        userDto.setId(String.valueOf(1)); // 外部传入的ID,通常与userName对应的ID一致或用于其他目的
        userDto.setName(new UserDTO.Name("surname", "firstname", "patronymic"));
        userDto.setActive(true);
        userDto.setEmails(List.of(new UserDTO.Email("email@example.com", "external")));
        userDto.setRoles(List.of(String.valueOf(roleId)));
        userDto.setLastAccessDate(LocalDateTime.of(2022, 10, 25, 4, 20));
        userDto.setUnit(null);

        // 2. 模拟依赖行为

        // 模拟 userRepository.findById() 返回一个已存在的UserEntity
        UserEntity existingUserEntity = new UserEntity();
        existingUserEntity.setId(userId); // 确保ID匹配
        existingUserEntity.setUserName(String.valueOf(userId));
        // ... 填充 existingUserEntity 的其他必要属性,使其在业务逻辑中有效 ...
        when(userRepository.findById(userId)).thenReturn(Optional.of(existingUserEntity));

        // 模拟 userService.mapToUserEntity() 的结果 (如果该方法是SUT的一部分,则不需要Mock)
        // 假设userService.mapToUserEntity是userService内部方法,直接调用即可
        UserEntity mappedUserEntity = userService.mapToUserEntity(userDto);
        // 如果mapToUserEntity是私有方法或难以直接调用,可以考虑Mock UserService本身或使用ArgumentCaptor

        // 模拟 userRepository.save() 返回保存后的UserEntity
        // 注意:这里返回的mappedUserEntity是期望被保存的实体,可能带上更新后的ID或版本
        when(userRepository.save(any(UserEntity.class))).thenReturn(mappedUserEntity);

        // 模拟 roleRepository.findById()
        when(roleRepository.findById(any())).thenReturn(Optional.of(new Object())); // 假设返回任意非空对象即可

        // 3. 调用被测方法
        UserDTO actualUserDto = userService.updateUser(String.valueOf(userDto.getId()), userDto);

        // 4. 验证结果
        assertNotNull(actualUserDto);
        // 验证返回的DTO是否符合预期,这里可以根据具体业务逻辑进行更细致的断言
        // 例如:assertEquals(userDto.getName().getFirstName(), actualUserDto.getName().getFirstName());
        // 如果userService.map方法会改变DTO,则不能直接比较userDto和actualUserDto
        // 简单起见,这里假设它们是等价的,或者至少某些关键属性是等价的
        assertEquals(userDto.getUserName(), actualUserDto.getUserName());
        assertEquals(userDto.getEmails().get(0).getValue(), actualUserDto.getEmails().get(0).getValue());

        // 验证 userRepository.findById 和 userRepository.save 是否被调用
        verify(userRepository).findById(userId);
        verify(userRepository).save(any(UserEntity.class));
    }

    // 假设UserService中存在这些辅助方法用于测试
    // 实际项目中,这些方法可能在UserService内部或独立的Mapper类中
    private static class UserService {
        private UserRepository userRepository;
        private RoleRepository roleRepository;

        // 构造函数用于注入Mock
        public UserService(UserRepository userRepository, RoleRepository roleRepository) {
            this.userRepository = userRepository;
            this.roleRepository = roleRepository;
        }

        // 模拟的mapToUserEntity方法
        public UserEntity mapToUserEntity(UserDTO dto) {
            UserEntity entity = new UserEntity();
            entity.setId(Integer.valueOf(dto.getUserName())); // 假设userName是ID
            entity.setFirstName(dto.getName().getFirstName());
            entity.setLastName(dto.getName().getLastName());
            entity.setEmail(dto.getEmails().get(0).getValue());
            entity.setActive(dto.isActive());
            entity.setLastAccessDate(dto.getLastAccessDate());
            // ... 其他属性映射 ...
            return entity;
        }

        // 模拟的map方法
        public UserDTO map(UserEntity entity) {
            UserDTO dto = new UserDTO();
            dto.setId(String.valueOf(entity.getId()));
            dto.setUserName(String.valueOf(entity.getId())); // 假设userName和ID一致
            dto.setName(new UserDTO.Name(entity.getLastName(), entity.getFirstName(), ""));
            dto.setActive(entity.isActive());
            dto.setEmails(List.of(new UserDTO.Email(entity.getEmail(), "external")));
            dto.setLastAccessDate(entity.getLastAccessDate());
            // ... 其他属性映射 ...
            return dto;
        }

        // 实际的updateUser方法,与问题描述中相同
        public UserDTO updateUser(String id, UserDTO updatedUser) {
            Optional databaseUser = userRepository.findById(Integer.valueOf(updatedUser.getUserName()));
            if (databaseUser.isEmpty()) {
                throw new UserNotFoundException("User with the this id is not found");
            }
            UserEntity entity = mapToUserEntity(updatedUser);
            return map(userRepository.save(entity));
        }
    }

    // 模拟的UserDTO类及其内部类
    private static class UserDTO {
        private String id;
        private String userName;
        private Name name;
        private boolean active;
        private List emails;
        private List roles;
        private LocalDateTime lastAccessDate;
        private Object unit; // 假设unit是Object类型

        // Getters and Setters
        public String getId() { return id; }
        public UserDTO setId(String id) { this.id = id; return this; }
        public String getUserName() { return userName; }
        public void setUserName(String userName) { this.userName = userName; }
        public Name getName() { return name; }
        public void setName(Name name) { this.name = name; }
        public boolean isActive() { return active; }
        public void setActive(boolean active) { this.active = active; }
        public List getEmails() { return emails; }
        public void setEmails(List emails) { this.emails = emails; }
        public List getRoles() { return roles; }
        public void setRoles(List roles) { this.roles = roles; }
        public LocalDateTime getLastAccessDate() { return lastAccessDate; }
        public void setLastAccessDate(LocalDateTime lastAccessDate) { this.lastAccessDate = lastAccessDate; }
        public Object getUnit() { return unit; }
        public void setUnit(Object unit) { this.unit = unit; }

        public static class Name {
            private String surname;
            private String firstname;

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
Golang 网络安全与加密实战
Golang 网络安全与加密实战

本专题系统讲解 Golang 在网络安全与加密技术中的应用,包括对称加密与非对称加密(AES、RSA)、哈希与数字签名、JWT身份认证、SSL/TLS 安全通信、常见网络攻击防范(如SQL注入、XSS、CSRF)及其防护措施。通过实战案例,帮助学习者掌握 如何使用 Go 语言保障网络通信的安全性,保护用户数据与隐私。

2

2026.01.29

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

446

2026.01.28

包子漫画在线官方入口大全
包子漫画在线官方入口大全

本合集汇总了包子漫画2026最新官方在线观看入口,涵盖备用域名、正版无广告链接及多端适配地址,助你畅享12700+高清漫画资源。阅读专题下面的文章了解更多详细内容。

145

2026.01.28

ao3中文版官网地址大全
ao3中文版官网地址大全

AO3最新中文版官网入口合集,汇总2026年主站及国内优化镜像链接,支持简体中文界面、无广告阅读与多设备同步。阅读专题下面的文章了解更多详细内容。

258

2026.01.28

php怎么写接口教程
php怎么写接口教程

本合集涵盖PHP接口开发基础、RESTful API设计、数据交互与安全处理等实用教程,助你快速掌握PHP接口编写技巧。阅读专题下面的文章了解更多详细内容。

9

2026.01.28

php中文乱码如何解决
php中文乱码如何解决

本文整理了php中文乱码如何解决及解决方法,阅读节专题下面的文章了解更多详细内容。

13

2026.01.28

Java 消息队列与异步架构实战
Java 消息队列与异步架构实战

本专题系统讲解 Java 在消息队列与异步系统架构中的核心应用,涵盖消息队列基本原理、Kafka 与 RabbitMQ 的使用场景对比、生产者与消费者模型、消息可靠性与顺序性保障、重复消费与幂等处理,以及在高并发系统中的异步解耦设计。通过实战案例,帮助学习者掌握 使用 Java 构建高吞吐、高可靠异步消息系统的完整思路。

10

2026.01.28

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

25

2026.01.27

拼多多赚钱的5种方法 拼多多赚钱的5种方法
拼多多赚钱的5种方法 拼多多赚钱的5种方法

在拼多多上赚钱主要可以通过无货源模式一件代发、精细化运营特色店铺、参与官方高流量活动、利用拼团机制社交裂变,以及成为多多进宝推广员这5种方法实现。核心策略在于通过低成本、高效率的供应链管理与营销,利用平台社交电商红利实现盈利。

124

2026.01.26

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
SQL 教程
SQL 教程

共61课时 | 3.6万人学习

10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号