
在单元测试中,应聚焦于被测类自身的逻辑行为而非外部服务实现;对 servicea 的测试需覆盖 serviceb 返回 true、false 和抛出异常三种场景,而非验证调用本身。
在单元测试中,应聚焦于被测类自身的逻辑行为而非外部服务实现;对 servicea 的测试需覆盖 serviceb 返回 true、false 和抛出异常三种场景,而非验证调用本身。
当 ServiceA 通过依赖注入调用 ServiceB.shouldVerify() 并仅透传其布尔返回值时,测试重点不是“是否调用了 ServiceB”,而是“ServiceA 如何响应不同返回结果”。这是因为:
- ServiceB 的业务逻辑已在它自己的测试套件中充分覆盖(如答案所述);
- ServiceA 本身不包含基于 shouldVerify() 返回值的分支判断(如 if (value) {...}),但只要该值参与后续流程(例如作为参数传递给其他服务、影响状态变更或触发日志/监控),其行为就具备可测性;
- 单元测试的核心原则是隔离被测对象:我们应使用 Mock(如 Mockito)模拟 ServiceB,主动控制其返回值与异常,从而验证 ServiceA.foo() 在各种契约边界下的表现。
✅ 正确的测试策略应覆盖以下三种典型场景:
- ServiceB.shouldVerify() 返回 true → 验证 ServiceA 后续逻辑是否按预期执行(如调用 serviceC.process(true));
- 返回 false → 验证对应路径(如调用 serviceC.process(false) 或跳过某步骤);
- 抛出异常(如 RuntimeException 或自定义 VerificationException) → 验证 ServiceA 是否妥善处理(如捕获、记录、转换异常或设置默认值)。
示例(使用 JUnit 5 + Mockito):
@ExtendWith(MockitoExtension.class)
class ServiceATest {
@Mock
private ServiceB serviceB;
@InjectMocks
private ServiceA serviceA;
@Test
void foo_shouldCallServiceCWithTrueWhenServiceBReturnsTrue() {
// given
when(serviceB.shouldVerify(any())).thenReturn(true);
// when
serviceA.foo();
// then
verify(serviceC).process(true); // 假设后续逻辑调用 serviceC.process(value)
}
@Test
void foo_shouldCallServiceCWithFalseWhenServiceBReturnsFalse() {
when(serviceB.shouldVerify(any())).thenReturn(false);
serviceA.foo();
verify(serviceC).process(false);
}
@Test
void foo_shouldHandleServiceBExceptionGracefully() {
when(serviceB.shouldVerify(any())).thenThrow(new RuntimeException("downstream failed"));
assertThrows(RuntimeException.class, () -> serviceA.foo());
// 或验证 fallback 行为,如:
// verify(logger).error(eq("Verification failed"), any(RuntimeException.class));
}
}⚠️ 注意事项:
- 避免测试“调用发生”本身(如 verify(serviceB).shouldVerify(...)),除非该调用是 ServiceA 的核心契约(例如:必须且仅能调用一次以保证幂等性)。本例中无此语义,故无需验证。
- 不要在 ServiceA 中重复测试 ServiceB 的逻辑(如 shouldVerify() 的具体条件判断),这属于测试职责错位,会导致冗余和脆弱测试。
- 若 ServiceA.foo() 当前确实完全忽略 value 的值(即未参与任何计算、未传递、未记录),则说明该调用存在设计冗余,应重构——要么移除无意义调用,要么补充业务逻辑使其可测。
总结:单元测试的价值在于保障被测类在协作环境中的行为确定性。对 ServiceA 而言,“是否调用 ServiceB”是实现细节;而“面对 ServiceB 的各种响应,ServiceA 是否健壮、可预测地完成自身职责”,才是测试必须回答的问题。










