为什么使用JUnit5

  • JUnit4被广泛使用,但是许多场景下使用起来语法较为繁琐,JUnit5中支持lambda表达式,语法简单且代码不冗余。
  • JUnit5易扩展,包容性强,可以接入其他的测试引擎。
  • 功能更强大提供了新的断言机制、参数化测试、重复性测试等新功能。
  • ps:开发人员为什么还要测试,单测写这么规范有必要吗?其实单测是开发人员必备技能,只不过很多开发人员开发任务太重导致调试完就不管了,没有系统化得单元测试,单元测试在系统重构时能发挥巨大的作用,可以在重构后快速测试新的接口是否与重构前有出入。

简介

structer

如图,JUnit5结构如下:

  • JUnit Platform: 这是Junit提供的平台功能模块,通过它,其它的测试引擎都可以接入Junit实现接口和执行。
  • JUnit JUpiter:这是JUnit5的核心,是一个基于JUnit Platform的引擎实现,它包含许多丰富的新特性来使得自动化测试更加方便和强大。
  • JUnit Vintage:这个模块是兼容JUnit3、JUnit4版本的测试引擎,使得旧版本的自动化测试也可以在JUnit5下正常运行。

依赖引入

我们以SpringBoot2.3.1为例,引入如下依赖,防止使用旧的junit4相关接口我们将其依赖排除。

        <dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
<exclusions>
<exclusion>
<groupId>org.junit.vintage</groupId>
<artifactId>junit-vintage-engine</artifactId>
</exclusion>
</exclusions>
</dependency>

常用注解

  • @BeforeEach:在每个单元测试方法执行前都执行一遍

  • @BeforeAll:在每个单元测试方法执行前执行一遍(只执行一次)

  • @DisplayName("商品入库测试"):用于指定单元测试的名称

  • @Disabled:当前单元测试置为无效,即单元测试时跳过该测试

  • @RepeatedTest(n):重复性测试,即执行n次

  • @ParameterizedTest:参数化测试,

  • @ValueSource(ints = {1, 2, 3}):参数化测试提供数据

断言

JUnit Jupiter提供了强大的断言方法用以验证结果,在使用时需要借助java8的新特性lambda表达式,均是来自org.junit.jupiter.api.Assertions包的static方法。

assertTrueassertFalse用来判断条件是否为truefalse

    @Test
@DisplayName("测试断言equals")
void testEquals() {
assertTrue(3 < 4);
}

assertNullassertNotNull用来判断条件是否为·null

    @Test
@DisplayName("测试断言NotNull")
void testNotNull() {
assertNotNull(new Object());
}

assertThrows用来判断执行抛出的异常是否符合预期,并可以使用异常类型接收返回值进行其他操作

    @Test
@DisplayName("测试断言抛异常")
void testThrows() {
ArithmeticException arithExcep = assertThrows(ArithmeticException.class, () -> {
int m = 5/0;
});
assertEquals("/ by zero", arithExcep.getMessage());
}

assertTimeout用来判断执行过程是否超时

    @Test
@DisplayName("测试断言超时")
void testTimeOut() {
String actualResult = assertTimeout(ofSeconds(2), () -> {
Thread.sleep(1000);
return "a result";
});
System.out.println(actualResult);
}

assertAll是组合断言,当它内部所有断言正确执行完才算通过

    @Test
@DisplayName("测试组合断言")
void testAll() {
assertAll("测试item商品下单",
() -> {
//模拟用户余额扣减
assertTrue(1 < 2, "余额不足");
},
() -> {
//模拟item数据库扣减库存
assertTrue(3 < 4);
},
() -> {
//模拟交易流水落库
assertNotNull(new Object());
}
);
}

重复性测试

在许多场景中我们需要对同一个接口方法进行重复测试,例如对幂等性接口的测试。

JUnit Jupiter通过使用@RepeatedTest(n)指定需要重复的次数

    @RepeatedTest(3)
    @DisplayName("重复测试")
    void repeatedTest() {
        System.out.println("调用");
    }

repeat

参数化测试

参数化测试可以按照多个参数分别运行多次单元测试这里有点类似于重复性测试,只不过每次运行传入的参数不用。需要使用到@ParameterizedTest,同时也需要@ValueSource提供一组数据,它支持八种基本类型以及String和自定义对象类型,使用极其方便。

    @ParameterizedTest
@ValueSource(ints = {1, 2, 3})
@DisplayName("参数化测试")
void paramTest(int a) {
assertTrue(a > 0 && a < 4);
}

内嵌测试

JUnit5提供了嵌套单元测试的功能,可以更好展示测试类之间的业务逻辑关系,我们通常是一个业务对应一个测试类,有业务关系的类其实可以写在一起。这样有利于进行测试。而且内联的写法可以大大减少不必要的类,精简项目,防止类爆炸等一系列问题。

@SpringBootTest
@AutoConfigureMockMvc
@DisplayName("Junit5单元测试")
public class MockTest {
//....
@Nested
@DisplayName("内嵌订单测试")
class OrderTestClas {
@Test
@DisplayName("取消订单")
void cancelOrder() {
int status = -1;
System.out.println("取消订单成功,订单状态为:"+status);
}
}
}

最新文章

  1. 11、Struts2 的文件上传和下载
  2. Bootstrap &lt;基础二十一&gt;徽章(Badges)
  3. FreeRTOS和Ucos在任务优先级的区别
  4. ui-router带参数的ui-sref配置
  5. vsftpd基本配置(原)
  6. 【HDOJ】3007 Buried memory
  7. java获取当前系统毫秒,纳秒
  8. COB (Chip On Board) 製程介紹/簡介/注意事項 I
  9. fzu Problem - 2232 炉石传说(二分匹配)
  10. 无线网络中,使用MDK3把指定的用户或者热点踢到掉线
  11. UNDO及MVCC、崩溃恢复
  12. Java 对IP请求进行限流.
  13. python全栈开发-Day13 内置函数
  14. Dos 批处理 Shutdown
  15. shell循环(两个日期比较,改变某个特定日期来改变当前比较值)
  16. django 简单路由配置
  17. day039 数据库索引
  18. 使用socket.io client 开发时兼容IE低版本的办法
  19. sql的函数和存储过程的区别
  20. nodejs API(一)

热门文章

  1. CF1349A Orac and LCM 题解
  2. RPC的实现的三种方式
  3. Min Cost Climbing Stairs [746]
  4. 终于有架构师用401页PDF+194张图表把设计模式讲明白了
  5. markdown 语法总结(一)
  6. 用LR录制终于可以在我的电脑上顺利弹出IE了
  7. dump 内存分析
  8. 《神经网络的梯度推导与代码验证》之vanilla RNN的前向传播和反向梯度推导
  9. Codeforces 1389 题解(A-E)
  10. 从String中移除空白字符的多种方式!?