
在单元测试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.java
@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.save(api);
return "recover";
}
}
// MyRestService.java
@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.java
@Service
public class MyStorageService {
@Autowired
MyRepo myRepo;
@Async
public MyEntity save(String api) {
System.out.println("Saving API: " + api + " to storage.");
return myRepo.save(new MyEntity(api, System.currentTimeMillis()));
}
}
// MyRepo.java (接口或抽象类)
public interface MyRepo {
MyEntity save(MyEntity entity);
}
// MyEntity.java
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()).save(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)).save(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);
}
}
}代码解释与改进点:
-
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对象,允许我们在测试中直接控制它们的行为。
-
正确使用when()和verify():
- when(mockRestService.call(eq(testApi), eq(testEntity))):现在我们对mockRestService这个Mock对象设置行为。eq()匹配器用于精确匹配参数值。
- deltaHelper.process(testApi, testEntity):调用SUT时,传入了真实的testApi和testEntity,而不是any()。
-
@EnableAspectJAutoProxy(proxyTargetClass = true):
- 这个注解确保Spring能够为带有@Retryable等AOP注解的类生成CGLIB代理(即使没有接口),从而使重试逻辑生效。
-
System.setProperty("delta.process.retries", "2"):
- 在setUp方法中设置系统属性,以确保@Retryable的maxAttemptsExpression = "${delta.process.retries}"能够正确解析重试次数。
-
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功能的组件进行单元测试,确保代码的健壮性和正确性。










