【单元测试】

单元测试的逻辑

有,而且很重要。单元测试虽然看起来是"每个类自己写",但其实有一套很稳定的通用逻辑。

最常见的套路就是这四步:

准备输入

准备依赖的假行为

执行被测试代码

断言结果和交互

很多地方会把它叫做 Arrange / Act / Assert,也就是:

Arrange:准备数据、准备 mock

Act:调用被测试方法

Assert:检查结果

拿你这个测试举例:

TencentMailClientFacade facade = mock(TencentMailClientFacade.class);

EmailNotificationSender sender = new EmailNotificationSender(facade);

这是在准备测试环境。

NotificationRequest request = new NotificationRequest();

...

when(facade.sendMail(request)).thenReturn("mail-msg-1");

这是在准备输入和依赖行为。

NotificationResponse response = sender.send(request);

这是执行被测方法。

verify(facade).sendMail(request);

assertEquals(...);

这是断言。

除了这个总套路,单元测试还有几个通用原则:

  1. 只测一个明确行为 不要一个测试里既测成功、又测失败、又测边界。
    一个测试最好只回答一个问题。
    比如:

    shouldReturnSentResponseWhenMailFacadeSucceeds

    shouldGenerateMessageIdWhenFacadeReturnsNull

    shouldPropagateExceptionWhenFacadeFails
    这样失败时你一眼就知道哪类行为坏了。
  2. 依赖要可控 单元测试里,数据库、Redis、MQ、HTTP、SMTP 这类外部依赖一般都 mock。
    不然测试会:



    不稳定

    受环境影响

    很难定位问题
  3. 断言两类东西 单元测试通常断言两种内容:

    状态结果

    返回值对不对

    字段对不对

    抛没抛异常

    交互行为

    有没有调用依赖

    调用了几次

    参数是不是对的
    你现在这个测试两种都断言了:

    assertEquals(...) 是状态断言

    verify(facade)... 是交互断言
  4. 覆盖正常路径、异常路径、边界路径 一个成熟的单元测试通常不会只测 happy path。
    一般会补三类:

    正常路径

    依赖成功,返回正常结果

    异常路径

    依赖抛异常时,你的代码怎么处理

    边界路径

    空值、默认值、特殊值、超长值等
    比如你这个 EmailNotificationSender,更完整一点至少会有:

    发送成功时返回正确响应

    facade 返回 null 时自动生成 messageId

    facade 抛异常时异常向上抛出
  5. 测"自己的逻辑",不要测框架 单元测试的重点不是测 Spring、Mockito、JDK,而是测你自己写的逻辑。
    比如:

    Spring 能不能注入 bean,这不是单元测试重点

    SMTP 能不能连通,这也不是单元测试重点

    你对结果的封装逻辑对不对,这才是重点
  6. 可重复、可快速执行 好的单元测试应该:

    本地随时能跑

    多次运行结果一致

    不依赖外网

    几秒内结束
    这也是为什么 mock 很常见。
    所以你可以把"单元测试的通用逻辑"总结成一句模板:
    给定一组输入和一组可控依赖,
    执行被测方法,
    断言返回结果、异常和依赖交互是否符合预期。
    如果再进一步落到你项目里,很多 service / facade / util 类的单元测试都可以套这个模版:
    // Arrange
    准备 mock
    准备输入
    预设依赖返回

// Act

调用被测方法

// Assert

断言返回值

断言异常

断言依赖调用

如果你愿意,我可以下一步直接拿你这个通知模块,给你列一个"单元测试清单",告诉你:

哪些类适合写单元测试

每个类至少该覆盖哪几种 case

哪些不该写成单元测试而应该写成集成测试

相关推荐
wb0430720112 小时前
厨房质检员——从阿明的“祖传配方“到标准化质检,看测试金字塔的落地
架构·log4j
汽车仪器仪表相关领域17 小时前
Kvaser Hybrid CAN/LIN 单通道三合一总线分析仪:高性价比CAN FD/LIN集成测试利器
运维·服务器·网络·数据挖掘·数据分析·单元测试·集成测试
一路往蓝-Anbo21 小时前
第十章:TDD部署 —— Ceedling 环境的深度集成
stm32·单片机·嵌入式硬件·单元测试·测试驱动开发·tdd
川石课堂软件测试2 天前
使用mock进行接口测试教程
数据库·python·功能测试·测试工具·华为·单元测试·appium
芒鸽2 天前
鸿蒙应用测试实战:从单元测试到自动化测试
华为·单元测试·harmonyos
MC皮蛋侠客3 天前
Google Test 单元测试指南
c++·单元测试·google test
英俊潇洒美少年5 天前
前端 Jest 单元测试零基础实战:模板、提效、避坑、面试题(Vue 项目可用)
前端·vue.js·单元测试
老码观察6 天前
设计模式实战解读(四):观察者模式——事件驱动的解耦利器
观察者模式·设计模式·log4j
TheRouter7 天前
LLM 应用的Evals 工程实践:从手动测试到自动化回归测试体系
运维·ai·自动化·log4j