新闻中心

正确测试Spring Retry组件:避免空指针与Mockito误用

2025-12-14
浏览次数:
返回列表

正确测试Spring Retry组件:避免空指针与Mockito误用

在单元测试spring retry功能时,开发者常遇到依赖注入为空或mockito用法不当的问题。本文将深入探讨如何正确配置spring测试环境,特别是如何有效模拟依赖,避免在测试系统核心逻辑时将真实对象误作模拟对象,以及如何规范使用`argumentmatchers.any()`,确保`@autowired`的bean能够正确注入,并使spring retry机制在测试中按预期工作。

理解Spring Retry的测试需求

Spring Retry通过AOP(面向切面编程)实现,这意味着它需要在Spring应用上下文中运行才能生效。因此,简单的JUnit测试无法激活其重试机制。我们需要借助Spring的测试工具,如SpringRunner(或Spring Boot的@SpringBootTest),并配置一个最小化的Spring应用上下文。

当使用@RunWith(SpringRunner.class)和@ContextConfiguration时,Spring会加载指定的配置类,并管理Bean的生命周期和依赖注入。然而,在测试中,我们往往需要隔离被测单元(System Under Test, SUT)的外部依赖,以便控制其行为并专注于SUT本身的逻辑。

常见测试陷阱与解决方案

在测试包含@Retryable注解的Spring组件时,开发者常会遇到以下两个主要问题:

陷阱一:将SUT误作Mock对象进行行为设置

问题描述: 开发者有时会尝试直接对被测类(SUT)的实例进行when()调用设置,例如 when(deltaHelper.restService.call(...)).thenThrow(...)。如果deltaHelper是真实的SUT实例,其内部的restService也是一个真实对象,那么when()方法将无法对其进行行为模拟,因为when()只能用于Mock对象。这会导致deltaHelper.restService`在测试执行时表现出真实行为,而不是我们期望的模拟行为,甚至可能因为未正确注入而导致空指针。

解决方案: 正确的做法是模拟SUT的依赖,而不是SUT本身。SUT应该是一个真实的Spring Bean,其依赖则应被替换为Mock对象。这样,我们可以在Mock依赖上设置期望的行为(例如抛出异常以触发重试),从而测试SUT在不同依赖行为下的响应。

陷阱二:在SUT的实际方法调用中使用ArgumentMatchers.any()

问题描述:ArgumentMatchers.any()(如any()、anyString()等)是Mockito提供的一种匹配器,用于在设置Mock行为(when())或验证Mock交互(verify())时匹配任何参数。然而,any()方法在被调用时会直接返回null(或对应基本类型的默认值)。如果在对SUT的实际方法调用中(即“act”阶段)使用any(),例如 deltaHelper.process(any(), any()),那么SUT接收到的参数将是null,这很可能导致空指针异常或其他非预期行为。

解决方案: 在调用SUT的实际方法时,必须传入真实的、有意义的参数值。any()仅应用于Mock对象的行为设置或验证。

优化Spring Retry组件的单元测试

结合上述解决方案,以下是针对DeltaHelper类的优化测试示例。我们将确保DeltaHelper是一个真实的Spring Bean,但其内部依赖MyRestService和MyStorageService将被替换为Mock对象。

首先,我们假设DeltaHelper、MyRestService和MyStorageService等业务组件已按常规Spring方式定义:

// DeltaHelper.j*a
@Component
public class DeltaHelper {

    @Autowired
    MyRestService restService;

    @Autowired
    MyStorageService myStorageService;

    @NotNull
    @Retryable(
            value = Exception.class,
            maxAttemptsExpression = "${delta.process.retries}"
    )
    public String process(String api, HttpEntity<?> entity) {
        System.out.println("Attempting process for API: " + api); // 方便观察重试
        return restService.call(api, entity);
    }

    @Recover
    public String recover(Exception e, String api, HttpEntity<?> entity) {
        System.out.println("Recovering from exception for API: " + api + " - " + e.getMessage());
        myStorageService.s*e(api);
        return "recover";
    }
}

// MyRestService.j*a
@Service
public class MyRestService extends org.springframework.web.client.RestTemplate {
    // 假设call方法存在并被DeltaHelper调用
    public String call(String api, HttpEntity<?> entity) {
        // 实际的REST调用逻辑
        throw new UnsupportedOperationException("Not implemented for real usage in test");
    }
}

// MyStorageService.j*a
@Service
public class MyStorageService {

    @Autowired
    MyRepo myRepo;

    @Async
    public MyEntity s*e(String api) {
        System.out.println("S*ing API: " + api + " to storage.");
        return myRepo.s*e(new MyEntity(api, System.currentTimeMillis()));
    }
}

// MyRepo.j*a (接口或抽象类)
public interface MyRepo {
    MyEntity s*e(MyEntity entity);
}

// MyEntity.j*a
public class MyEntity {
    private String api;
    private long timestamp;

    public MyEntity(String api, long timestamp) {
        this.api = api;
        this.timestamp = timestamp;
    }
    // getters, setters
}

接下来是修正后的测试类:

import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.Mockito;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import org.springframework.http.HttpEntity;
import org.springframework.retry.annotation.EnableRetry;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringRunner;

import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.*;

@RunWith(SpringRunner.class)
@ContextConfiguration
public class DeltaHelperTest {

    @Autowired
    private DeltaHelper deltaHelper; // SUT,由Spring管理并注入Mock依赖

    @Autowired
    private MyRestService mockRestService; // 注入Mock对象,用于设置行为和验证

    @Autowired
    private MyStorageService mockMyStorageService; // 注入Mock对象

    @Autowired
    private MyRepo mockMyRepo; // 注入Mock对象

    @Before
    public void setUp() {
        // 设置重试次数,确保@Retryable的maxAttemptsExpression能正确解析
        System.setProperty("delta.process.retries", "2");
        // 重置所有Mock,确保每个测试方法都是干净的环境
        Mockito.reset(mockRestService, mockMyStorageService, mockMyRepo);
    }

    @After
    public void validate() {
        // 验证Mock的使用,确保没有未验证的交互
        validateMockitoUsage();
    }

    @Test
    public void retriesAfterOneFailAndThenPass() throws Exception {
        String testApi = "test-api-path";
        HttpEntity<?> testEntity = new HttpEntity<>("test-body");

        // 模拟restService的第一次调用抛出异常,第二次成功
        when(mockRestService.call(eq(testApi), eq(testEntity)))
                .thenThrow(new RuntimeException("Simulated first call failure")) // 第一次失败
                .thenReturn("success-response"); // 第二次成功

        // 调用SUT的方法,传入真实的参数
        String result = deltaHelper.process(testApi, testEntity);

        // 验证restService的call方法被调用了两次(一次失败,一次成功)
        verify(mockRestService, times(2)).call(eq(testApi), eq(testEntity));
        // 验证重试成功后返回的是第二次调用的结果
        assert "success-response".equals(result);
        // 验证recover方法没有被调用,因为重试成功了
        verify(mockMyStorageService, never()).s*e(anyString());
    }

    @Test
    public void retriesFailAndThenRecover() throws Exception {
        String testApi = "fail-api-path";
        HttpEntity<?> testEntity = new HttpEntity<>("fail-body");

        // 模拟restService的两次调用都抛出异常,触发recover
        when(mockRestService.call(eq(testApi), eq(testEntity)))
                .thenThrow(new RuntimeException("Simulated first call failure"))
                .thenThrow(new RuntimeException("Simulated second call failure")); // 第二次也失败

        // 调用SUT的方法,传入真实的参数
        String result = deltaHelper.process(testApi, testEntity);

        // 验证restService的call方法被调用了两次(达到maxAttemptsExpression设定的次数)
        verify(mockRestService, times(2)).call(eq(testApi), eq(testEntity));
        // 验证recover方法被调用了,因为重试失败
        verify(mockMyStorageService, times(1)).s*e(eq(testApi));
        // 验证返回的是recover方法的结果
        assert "recover".equals(result);
    }

    @Configuration
    @EnableRetry // 启用Spring Retry
    @EnableAspectJAutoProxy(proxyTargetClass = true) // 启用AspectJ代理,确保@Retryable生效
    public static class Application {

        // DeltaHelper作为SUT,让Spring正常创建和注入
        @Bean
        public DeltaHelper deltaHelper() {
            return new DeltaHelper();
        }

        // 提供MyRestService的Mock Bean
        @Bean
        public MyRestService restService() {
            return mock(MyRestService.class);
        }

        // 提供MyStorageService的Mock Bean
        @Bean
        public MyStorageService myStorageService() {
            return mock(MyStorageService.class);
        }

        // 提供MyRepo的Mock Bean
        @Bean
        public MyRepo myRepository() {
            return mock(MyRepo.class);
        }
    }
}

代码解释与改进点:

挖错网 挖错网

一款支持文本、图片、视频纠错和AIGC检测的内容审核校对平台。

挖错网 185 查看详情 挖错网
  1. SUT作为真实Bean,依赖作为Mock Bean:

    • 在Application配置类中,deltaHelper()方法现在直接返回 new DeltaHelper()。由于DeltaHelper类上带有@Component注解,Spring会扫描并将其作为Bean管理。
    • restService()、myStorageService() 和 myRepository() 方法现在返回 mock(...) 创建的Mock对象。这些Mock对象会被注入到DeltaHelper中。
    • @Autowired DeltaHelper deltaHelper; 会注入由Spring创建的真实DeltaHelper实例。
    • @Autowired MyRestService mockRestService; 等会注入我们定义的Mock对象,允许我们在测试中直接控制它们的行为。
  2. 正确使用when()和verify():

    • when(mockRestService.call(eq(testApi), eq(testEntity))):现在我们对mockRestService这个Mock对象设置行为。eq()匹配器用于精确匹配参数值。
    • deltaHelper.process(testApi, testEntity):调用SUT时,传入了真实的testApi和testEntity,而不是any()。
  3. @EnableAspectJAutoProxy(proxyTargetClass = true):

    • 这个注解确保Spring能够为带有@Retryable等AOP注解的类生成CGLIB代理(即使没有接口),从而使重试逻辑生效。
  4. System.setProperty("delta.process.retries", "2"):

    • 在setUp方法中设置系统属性,以确保@Retryable的maxAttemptsExpression = "${delta.process.retries}"能够正确解析重试次数。
  5. Mockito.reset(...):

    • 在setUp中重置所有Mock对象,确保每个测试方法都在一个干净的状态下运行,避免测试之间的状态污染。

注意事项与总结

  • 测试边界: 单元测试应聚焦于SUT的逻辑,而不是Spring框架本身的功能。我们假定@Retryable注解本身是正确的,我们测试的是当依赖抛出异常时,SUT的重试逻辑是否按预期执行。
  • 清晰的职责分离: 明确哪些是SUT,哪些是SUT的依赖。SUT是我们要测试的核心业务逻辑,它应该是真实的。依赖则是我们为了隔离SUT而需要模拟的部分。
  • 有意义的测试数据: 避免在SUT的实际调用中使用any()。提供具体的、有意义的参数,使测试场景更真实,也更容易调试。
  • Spring Boot Test的便利性: 如果项目是Spring Boot应用,可以使用@SpringBootTest结合@MockBean来更简洁地替换依赖。@MockBean会自动将指定的类替换为Mock对象并注入到Spring上下文中。

通过遵循这些原则和实践,您可以更有效地对包含Spring Retry功能的组件进行单元测试,确保代码的健壮性和正确性。

以上就是正确测试Spring Retry组件:避免空指针与Mockito误用的详细内容,更多请关注其它相关文章!


# 有意义  # 百度关键词排名批量查询  # 网站推广能干啥呢知乎  # 网站优化公司排名深圳  # 铜川seo公司方便火星  # 揭阳网站推广你  # 郑州网站建设管理  # 研发网站建设推广  # 现代服饰网站建设思路  # 济南搜狗关键词排名  # 蚌埠网站推广设计招聘  # 测试中  # 转换为  # 单元测试  # java  # 是一个  # 而不是  # 两次  # 抛出  # 的是  # 重试  # red  # spring框架  # springboot  # proxy  # ai  # 工具  # app 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: PHP表单数据传递:如何通过隐藏输入字段获取动态ID  Go语言HTML解析:利用Goquery精准获取指定元素内容  c++项目目录结构应该如何组织_c++工程化项目结构规范  如何在网页中实现特定地点的随机图片展示  Python异步编程实践:使用Binance API构建实时交易数据流  Pandas DataFrame 多条件优先级排序与排名  Golang如何使用buffered channel提高性能_Golang buffered channel优化技巧  mysql如何设置表访问权限_mysql表访问权限配置  J*aScriptWebpack优化_J*aScript构建工具实战  妖精漫画网页版登录入口免费_妖精漫画官网主页直接阅读漫画  Python:递归比较文件夹内容并找出特定类型文件的差异  在J*a中如何开发简易博客标签推荐系统_博客标签推荐项目实战解析  Spring Boot内嵌服务器与J*a EE全栈特性:选择与部署策略  Mac怎么查看崩溃日志_Mac控制台错误报告分析  KFC早餐时段怎么领特惠代码_KFC早餐订餐优惠代码获取与使用说明  Lar*el表单中优雅地处理“返回”按钮以规避验证:最佳实践指南  steam官方网页快速访问 steam账号注册全流程  俄罗斯Yandex免登录入口_Yandex搜索引擎官网一键直达  深入理解J*a编译器的兼容性选项:从-source到--release  C++如何打印当前代码行号与文件名_C++预定义宏FILE与LINE的使用  Yandex官网免登录入口_俄罗斯Yandex搜索引擎一键访问  Win10怎么设置静态IP地址 Win10手动配置IP地址步骤【指南】  Python实现多节点属性重叠度分析教程  Python字典中优雅地迭代剩余元素的方法  J*a如何使用AtomicInteger控制计数_J*a无锁计数器性能分析  痛风发作了怎么办? 快速止痛和后期饮食调理  微信商城在哪里打开【步骤】  俄罗斯浏览器官网直达链接 俄罗斯浏览器最新在线入口导航  必由学官网首页入口 必由学教师网页版登录指南  三星ZFold5多任务卡顿_Samsung ZFold5流畅度提升  126邮箱账号注册 电脑版登录入口  理解J*aScript Promise的微任务队列与执行顺序  钉钉视频会议画面卡顿如何解决 钉钉会议画面优化方法  CKEditor 5 自定义构建在React应用中渲染失败的调试与解决  LINUX的perf命令入门_LINUX官方性能分析工具的使用与解读  抖音创作助手登录入口_抖音创作辅助工具官网直达  向日葵客户端怎么进行远程CentOS控制_向日葵客户端远程CentOS控制操作教程  谷歌google账号注册详细步骤 谷歌账号注册官方教程  在命令行怎么运行html项目_命令行运行html项目方法【教程】  b站如何看历史记录_b站观看历史找回方法  微博网页版官方账号登录 微博网页版内容浏览使用指南  深入理解字体排版:Adobe光学字偶距与CSS字偶距的差异与实现  如何在复杂的电商平台中优雅地管理共享资源并确保正确重定向,使用spryker-shop/resource-share-page模块助你一臂之力  马斯克:Optimus 人形机器人复数形式为 Optimi  微信聊天记录怎么加密_微信聊天记录加密方法  抖音怎么赚钱_抖音创作者变现方法与途径指南  Golang如何实现状态模式管理对象状态_Golang State模式实现技巧  Win11怎么开启省电模式_Win11电池节电模式自动开启  蓝湖怎样用切图标注提对接效率_蓝湖用切图标注提对接效率【设计对接】  拼多多视频播放卡顿如何处理 拼多多视频播放优化技巧 

搜索