
Mockito 创建的 mock 对象不调用真实构造函数,也不保存真实字段状态;所有未 stub 的方法默认返回零值(如 0.0、null、false),因此 getPrice() 返回 0.0 而非构造中硬编码的 18.99。
mockito 创建的 mock 对象不调用真实构造函数,也不保存真实字段状态;所有未 stub 的方法默认返回零值(如 `0.0`、`null`、`false`),因此 `getprice()` 返回 `0.0` 而非构造中硬编码的 `18.99`。
在使用 Mockito 进行单元测试时,一个常见误区是认为 @Mock 注解会创建一个“部分初始化”的真实对象——例如,期望 Football 的 mock 实例能记住其父类构造器中传入的 18.99 并在 getPrice() 中返回该值。但事实并非如此。
Mock 对象的本质是行为替身,而非轻量实例化。
当你声明:
@Mock Football football;
Mockito 并不会调用 Football 的任何构造函数(包括 super(colour, 18.99)),也不会初始化其父类 Item 中的 price 字段。相反,它通过字节码增强技术生成一个代理类,其中所有 public/non-final 方法均被重写为默认存根逻辑(default answer)。根据 Mockito 官方文档,默认返回值规则如下:
- double, int, boolean 等基本类型 → 返回对应零值(0.0, 0, false)
- Double, Integer, Boolean 等包装类型 → 返回 null(⚠️注意:此处需特别澄清——实际行为取决于 Mockito 版本与配置;自 Mockito 2.0+ 默认对包装类型也返回 null,但 Double 类型的 getPrice() 在多数版本中因返回类型可空且未显式 stub,常被误观察为 0.0,实为 null 导致 NPE 或自动拆箱为 0.0)
验证这一点的最直接方式是添加断言:
@Test
void mock_getPrice_returnsNullByDefault() {
// 注意:此处 football 是纯 mock,无构造逻辑执行
Double result = football.getPrice(); // 实际返回 null(非 18.99)
assertThat(result).isNull(); // ✅ 通过(若未 stub)
}✅ 正确做法:按需显式 stub 行为
若你确实需要 getPrice() 返回 18.99,必须主动定义:
@BeforeEach
void initMocks() {
MockitoAnnotations.openMocks(this);
// 显式指定 getPrice() 的返回值
when(football.getPrice()).thenReturn(18.99);
}或者使用 @Mock(answer = Answers.RETURNS_DEEP_STUBS)(不推荐用于简单 getter)、@Spy(适用于真实对象部分模拟),但需注意:
-
@Spy 会调用真实构造函数,适合此场景:
@Spy Football football = new Football(Colour.RED, 100.0); // 构造执行,price 初始化为 18.99 // 此时 getPrice() 默认返回 18.99,无需 stub assertThat(football.getPrice()).isEqualTo(18.99);
⚠️ 关键注意事项:
- @Mock ≠ “带初始值的对象”,它完全脱离原始类的生命周期;
- 不要依赖 mock 的字段值或构造副作用——mock 的唯一职责是可控地响应方法调用;
- 若业务逻辑高度依赖构造参数和字段状态,优先考虑 @Spy、真实对象测试,或重构为更易测的设计(如将价格计算提取为独立服务);
- 使用 Mockito.verifyNoInteractions(football) 可确认 mock 是否被意外调用,辅助调试行为预期。
总之,理解 mock 与 spy 的根本区别,是写出可靠、可维护测试的基础。








