(三)、SpringBoot 核心功能
4.单元测试功能
(1).JUnit5 的变化
Spring Boot 2.2.0 版本开始引入 JUnit 5 作为单元测试默认库
作为最新版本的JUnit框架,JUnit5与之前版本的Junit框架有很大的不同。由三个不同子项目的几个不同模块组成。
JUnit 5 = JUnit Platform + JUnit Jupiter + JUnit Vintage
- JUnit Platform: Junit Platform是在JVM上启动测试框架的基础,不仅支持Junit自制的测试引擎,其他测试引擎也都可以接入。
- JUnit Jupiter: JUnit Jupiter提供了JUnit5的新的编程模型,是JUnit5新特性的核心。内部 包含了一个测试引擎,用于在Junit Platform上运行。
- JUnit Vintage: 由于JUint已经发展多年,为了照顾老的项目,JUnit Vintage提供了兼容JUnit4.x,Junit3.x的测试引擎。
注意:
SpringBoot 2.4 以后版本移除了默认对 Vintage 的依赖。如果需要兼容junit4需要自行引入(不能使用junit4的功能 @Test)
JUnit 5’s Vintage Engine Removed from spring-boot-starter-test
,如果需要继续兼容junit4需要自行引入vintage.
<dependency> <groupId>org.junit.vintage</groupId> <artifactId>junit-vintage-engine</artifactId> <scope>test</scope> <exclusions> <exclusion> <groupId>org.hamcrest</groupId> <artifactId>hamcrest-core</artifactId> </exclusion> </exclusions> </dependency>
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency>
@SpringBootTest class Boot05WebAdminApplicationTests { @Test void contextLoads() { } }
以前:
@SpringBootTest + @RunWith(SpringTest.class)
SpringBoot整合Junit以后
- 编写测试方法:
@Test标注
(注意需要使用junit5版本的注解) - Junit类具有Spring的功能,@Autowired、比如 @Transactional 标注测试方法,测试完成后自动回滚
(2).JUnit5常用注解
JUnit5的注解与JUnit4的注解有所变化
https://junit.org/junit5/docs/current/user-guide/#writing-tests-annotations
- @Test :表示方法是测试方法。但是与JUnit4的@Test不同,他的职责非常单一不能声明任何属性,拓展的测试将会由Jupiter提供额外测试
- @ParameterizedTest :表示方法是参数化测试,下方会有详细介绍
- @RepeatedTest :表示方法可
重复执行(重复执行几次)
,下方会有详细介绍 - @DisplayName :为测试类或者测试方法
设置展示名称
- @BeforeEach :表示在
每个单元测试
之前执行 - @AfterEach :表示在
每个单元测试
之后执行 - @BeforeAll :表示在
所有单元测试
之前执行 - @AfterAll :表示在
所有单元测试
之后执行 - @Tag :表示单元测试类别,类似于JUnit4中的@Categories
- @Disabled :表示
测试类或测试方法不执行
,类似于JUnit4中的@Ignore - @Timeout :表示
测试方法运行如果超过了指定时间将会返回错误
- @ExtendWith :为测试类或测试方法提供
扩展类引用
.(@springBootTest注解已经包含)
- @DisplayName
package com.jsxs; import com.jsxs.bean.account; import org.junit.jupiter.api.DisplayName; import org.junit.jupiter.api.Test; import org.springframework.boot.test.context.SpringBootTest; import org.springframework.jdbc.core.JdbcTemplate; import javax.annotation.Resource; import java.util.List; import java.util.Map; @DisplayName("测试Juint5") @SpringBootTest class SpringBootLs02ApplicationTests { @DisplayName("测试Display") @Test void test(){ System.out.println(1); } }
- BeforeEach 和 AfterEach
package com.jsxs; import com.jsxs.bean.account; import org.junit.jupiter.api.*; import org.springframework.boot.test.context.SpringBootTest; import org.springframework.jdbc.core.JdbcTemplate; import javax.annotation.Resource; import java.util.List; import java.util.Map; @DisplayName("测试Juint5") @SpringBootTest class SpringBootLs02ApplicationTests { @DisplayName("测试Display") @Test void test(){ System.out.println(1); } @BeforeEach void testBefore(){ System.out.println("在单元测试之前执行..."); } @AfterEach void testAfter(){ System.out.println("在单元测试之后执行..."); }
- BeforeAll 和 AfterAll (这里的方法都是静态方法)
package com.jsxs; import com.jsxs.bean.account; import org.junit.jupiter.api.*; import org.springframework.boot.test.context.SpringBootTest; import org.springframework.jdbc.core.JdbcTemplate; import javax.annotation.Resource; import java.util.List; import java.util.Map; @DisplayName("测试Juint5") @SpringBootTest class SpringBootLs02ApplicationTests { @DisplayName("测试Display") @Test void test(){ System.out.println(1); } @BeforeEach void testBefore(){ System.out.println("在单元测试之前执行..."); } @AfterEach void testAfter(){ System.out.println("在单元测试之后执行..."); } @BeforeAll // 需要标注静态方法 static void testALLBefore(){ System.out.println("在--所有单元--测试之前执行..."); } @AfterAll // 需要标注静态方法 static void testALLAfter(){ System.out.println("在--所有单元--测试之后执行..."); } }
(3).断言( assertions
)
断言(assertions)是测试方法中的核心部分,用来对测试需要满足的条件进行验证
。这些断言方法都是 org.junit.jupiter.api.Assertions 的静态方法
。JUnit 5 内置的断言可以分成如下几个类别:检查业务逻辑返回的数据是否合理。
所有的测试运行结束以后,会有一个详细的测试报告;
(3.1).简单断言
只要前面的断言有一个失败的,后面的都不会执行。
用来对单个值进行简单的验证。如:
@DisplayName("测试简单断言...") @Test public void simple(){ int add = add(1, 2); // 使用断言的操作 Assertions.assertEquals(4,add,"业务逻辑与预期值不一致"); assertNotSame(new Object(), new Object()); Object obj = new Object(); assertSame(obj, obj); assertFalse(1 > 2); assertTrue(1 < 2); assertNull(null); assertNotNull(new Object()); } int add(int x,int y){ return x+y; }
(3.2).数组断言
通过 assertArrayEquals
方法来判断两个对象或原始类型的数组是否相等
@Test @DisplayName("array assertion") public void array() { assertArrayEquals(new int[]{1, 2}, new int[] {1, 2}); }
(3.3).组合断言
assertAll 方法接受多个 org.junit.jupiter.api.Executable 函数式接口
的实例作为要验证的断言,可以通过 lambda 表达式很容易的提供这些断言.
@Test @DisplayName("assert all") public void all() { assertAll("Math", () -> assertEquals(2, 1 + 1), () -> assertTrue(1 > 0) ); }
(3.4).异常断言
在JUnit4时期,想要测试方法的异常情况时,需要用@Rule
注解的ExpectedException变量还是比较麻烦的。而JUnit5提供了一种新的断言方式Assertions.assertThrows()
,配合函数式编程就可以进行使用。
@Test @DisplayName("异常测试") public void exceptionTest() { ArithmeticException exception = Assertions.assertThrows( //扔出断言异常 ArithmeticException.class, () -> System.out.println(1 % 0)); }
(3.5).超时断言
Junit5还提供了Assertions.assertTimeout() 为测试方法设置了超时时间.
@Test @DisplayName("超时测试") public void timeoutTest() { //如果测试方法时间超过1s将会异常 Assertions.assertTimeout(Duration.ofMillis(1000), () -> Thread.sleep(500)); }
(3.6).快速失败
通过 fail 方法直接使得测试失败
@Test @DisplayName("fail") public void shouldFail() { fail("This should fail"); }
(4).前置条件( assumptions
)
JUnit 5 中的前置条件(assumptions
【假设】)类似于断言,不同之处在于不满足的断言会使得测试方法失败,而不满足的前置条件只会使得测试方法的执行终止。前置条件可以看成是测试方法执行的前提,当该前提不满足时,就没有继续执行的必要。
@DisplayName("前置条件") @Test public void testDuan(){ Assumptions.assumeTrue(false,"结果不是true"); System.out.println(111); }
所有的测试标志汇总: