Spring Boot 单元测试:@SpyBean 使用教程

Spring Boot 单元测试:@SpyBean 使用教程

1. 什么是 @SpyBean

@SpyBean 是 Spring Boot Test 提供的一个注解,用于在单元测试中**部分模拟(Partial Mock)**一个 Spring Bean。它类似于 Mockito 的 @Spy,但专门用于 Spring 容器中的 Bean。

  • @MockBean:完全模拟一个 Bean,所有方法都需要手动定义行为。
  • @SpyBean:基于真实 Bean,只重写(Mock)部分方法,其余方法仍然调用真实逻辑。

2. 何时使用 @SpyBean

适用场景

  • 你只想修改某个 Bean 的某个方法,但其他方法仍然走真实逻辑。
  • 你希望测试依赖某个 Bean 的真实逻辑,但需要控制其中某个方法的返回值。

不适用场景

  • 需要完全模拟整个 Bean(用 @MockBean)。
  • Bean 的方法是 finalstatic(Mockito 无法 Spy final 方法)。

3. 基本用法

示例场景

假设有一个 UserService 依赖 UserRepository,我们想测试 UserService,但只重写 UserRepositoryfindById() 方法,其他方法(如 save())仍然走真实逻辑。

java 复制代码
@Service
public class UserService {
    @Autowired
    private UserRepository userRepository;

    public String getUserName(Long id) {
        User user = userRepository.findById(id).orElseThrow();
        return user.getName();
    }
}

测试代码

java 复制代码
@SpringBootTest
public class UserServiceTest {

    @Autowired
    private UserService userService;

    @SpyBean  // 对 UserRepository 进行 Spy,未 Mock 的方法仍然走真实逻辑
    private UserRepository userRepository;

    @Test
    public void testGetUserName() {
        // 准备测试数据
        User mockUser = new User(1L, "Alice");
        userRepository.save(mockUser);  // 真实调用 save()

        // 只 Mock findById(),让它返回自定义数据
        doReturn(Optional.of(new User(1L, "Mocked Alice")))
            .when(userRepository).findById(1L);

        // 调用测试方法
        String userName = userService.getUserName(1L);

        // 验证
        assertEquals("Mocked Alice", userName);

        // 其他方法仍然走真实逻辑
        User savedUser = userRepository.findById(2L).orElse(null);
        assertNull(savedUser);  // 因为没有存 ID=2 的用户
    }
}

4. @SpyBean 的常用 Mock 方式

(1) doReturn().when(spy).method()

java 复制代码
// 让方法返回固定值
doReturn("Mocked Result").when(mySpyBean).someMethod();

// 带参数匹配
doReturn("Hello").when(mySpyBean).greet(anyString());

(2) doAnswer()(自定义逻辑)

java 复制代码
doAnswer(invocation -> {
    String arg = invocation.getArgument(0);
    return "Processed: " + arg;
}).when(mySpyBean).process(anyString());

(3) doThrow()(模拟异常)

java 复制代码
doThrow(new RuntimeException("DB Error"))
    .when(mySpyBean).save(any());

(4) doCallRealMethod()(显式调用真实方法)

java 复制代码
// 默认会走真实方法,但也可以显式声明
doCallRealMethod().when(mySpyBean).someMethod();

5. @SpyBean vs @MockBean

特性 @SpyBean @MockBean
默认行为 未 Mock 的方法调用真实逻辑 未 Mock 的方法返回默认值(null/0/false)
适用场景 只修改部分方法,其他方法仍需真实逻辑 完全模拟 Bean,不依赖真实实现
性能 稍慢(需代理真实对象) 更快(纯 Mock 对象)

6. 常见问题

Q1: @SpyBean 能用于 final 方法吗?

不能! Mockito 无法 Spy final 方法。如果遇到 final 方法,考虑:

  • 改用 @MockBean 完全模拟。
  • 重构代码,避免 final 方法。

Q2: @SpyBean@Spy 的区别?

  • @Spy 是 Mockito 提供的,用于普通对象。
  • @SpyBean 是 Spring Boot 提供的,用于 Spring Bean。

Q3: 如何验证方法调用次数?

java 复制代码
// 验证 someMethod 被调用 1 次
verify(mySpyBean, times(1)).someMethod();

7. 总结

  • @SpyBean 适合部分 Mock,只修改某些方法,其余方法仍然走真实逻辑。
  • 常用方式
    • doReturn().when(spy).method():返回固定值。
    • doAnswer():自定义逻辑。
    • doThrow():模拟异常。
  • 不要用于 final 方法

🚀 现在你可以在 Spring Boot 测试中灵活使用 @SpyBean 了! 如果有问题,欢迎讨论!

相关推荐
四问四不知2 小时前
Understand Anything的初步了解
log4j
终将老去的穷苦程序员8 小时前
基于SpringBoot的餐饮管理系统
java·spring boot·后端
心之伊始8 小时前
Spring AI Tool Calling 实战:让 Java Agent 调用本地 Bean 工具方法
java·spring boot·agent·spring ai·tool calling
熠熠仔8 小时前
Spring Boot 与 MyBatis-Plus 空间几何数据集成指南
spring boot·后端·mybatis
HLAIA光子9 小时前
AI Coding框架,打好TDD和SDD这两拳
单元测试·ai编程·代码规范
天青色等烟雨..9 小时前
智慧农林核心遥感技术99个案例实践
运维·人工智能·spring boot·后端·自动化
雪宫街道11 小时前
SpringBoot 向 IOC 容器注册组件的两种姿势:@Configuration 与 @Import
java·spring boot·后端·spring
接着奏乐接着舞12 小时前
springboot mp mybatis plaus
windows·spring boot·mybatis
金融支付架构实战指南12 小时前
Milvus 向量检索服务 + SpringBoot 实战:电商商品语义检索与相似商品推荐
spring boot·后端·milvus·向量检索
愛~杦辷个訾13 小时前
Java Springboot使用阿里云oss对图片进行等质量压缩,转换成webp格式的压缩图。
java·spring boot·阿里云·oss