当前位置:   article > 正文

【单元测试】如何使用 JUnit5 框架?

junit5

JUnit5 单元测试框架使用教程

一、Junit5 是什么?

  Junit5是一个用于在Java平台上进行单元测试的框架。JUnit 5 框架主要由三部分组成:JUnit Platform、JUnit Jupiter 和 JUnit Vintage。

  • JUnit Platform:定义了测试引擎的 API,是 JVM 上用于启动测试框架的基础服务,支持通过 IDE、构建工具、命令行等方式运行单元测试。
  • JUnit Jupiter:包含 JUnit 5 新的编程模型和扩展模型,主要用于编写和扩展测试代码。
  • JUnit Vintage:兼容运行 JUnit 3 和 JUnit4 编写的测试用例。

二、Junit5 的注解

(一)导入依赖

  导入五个依赖:

  1. <!-- junit-jupiter-api 里有 @BeforeAll……等注解 -->
  2. <dependency>
  3. <groupId>org.junit.jupiter</groupId>
  4. <artifactId>junit-jupiter-api</artifactId>
  5. <version>5.10.0</version>
  6. </dependency>
  7. <!-- junit-jupiter-params 里有 @ValueSource……等注解 -->
  8. <dependency>
  9. <groupId>org.junit.jupiter</groupId>
  10. <artifactId>junit-jupiter-params</artifactId>
  11. <version>5.10.0</version>
  12. </dependency>
  13. <!-- 用于运行 识别上述注解 -->
  14. <dependency>
  15. <groupId>org.junit.jupiter</groupId>
  16. <artifactId>junit-jupiter-engine</artifactId>
  17. <version>5.10.0</version>
  18. <scope>test</scope>
  19. </dependency>
  20. <!-- 测试套件 -->
  21. <dependency>
  22. <groupId>org.junit.platform</groupId>
  23. <artifactId>junit-platform-suite-api</artifactId>
  24. <version>1.10.0</version>
  25. </dependency>
  26. <!-- 运行测试套件的测试引擎-->
  27. <dependency>
  28. <groupId>org.junit.platform</groupId>
  29. <artifactId>junit-platform-suite-engine</artifactId>
  30. <version>1.10.0</version>
  31. <scope>test</scope>
  32. </dependency>

(二)常用的注解

如果你的IDEA在使用JUnit注解的时候发生如下情况:依赖已经导入且加载完成,但是IDEA没能识别出来,如图:

image.png

有的注解也在params包中。

image.png

(我真的不理解为啥)

有一种解决办法:

image.png

image.png

image.png

点击后,选择相应的版本,我这里是5.10.0,点了之后IDEA就能识别出来了。

  1. 现在我也找了很多测试的朋友,做了一个分享技术的交流群,共享了很多我们收集的技术文档和视频教程。
  2. 如果你不想再体验自学时找不到资源,没人解答问题,坚持几天便放弃的感受
  3. 可以加入我们一起交流。而且还有很多在自动化,性能,安全,测试开发等等方面有一定建树的技术大牛
  4. 分享他们的经验,还会分享很多直播讲座和技术沙龙
  5. 可以免费学习!划重点!开源的!!!
  6. qq群号:110685036

1.@Test
  1. public class JUnitTest {
  2. @Test
  3. void test0(){
  4. System.out.println("测试用例1");
  5. }
  6. @Test
  7. void test1(){
  8. System.out.println("测试用例2");
  9. }
  10. @Test
  11. void test2(){
  12. System.out.println("测试用例3");
  13. }
  14. }

结果:

image.png

2.@BeforeAll、@AfterAll
  • @BeforeAll:表示被注解的方法应该在当前类的所有@Test,@RepeatedTest,@ParameterizedTest和@TestFactory方法之前执行;
  • @AfterAll:表示被注解的方法应该在当前类的所有@Test,@RepeatedTest,@ParameterizedTest和@TestFactory方法之后执行;

使用@BeforeAll@AfterAll注解的方法要加上static

  1. public class JUnitTest {
  2. @BeforeAll
  3. static void beforeAll(){
  4. //可以用于创建一些资源
  5. System.out.println("我是BeforeAll,我最开始执行。");
  6. }
  7. @AfterAll
  8. static void afterAll(){
  9. //可以用于释放资源
  10. System.out.println("我是AfterAll,我最后执行。");
  11. }
  12. @Test
  13. void test0(){
  14. System.out.println("测试用例1");
  15. }
  16. @Test
  17. void test1(){
  18. System.out.println("测试用例2");
  19. }
  20. @Test
  21. void test2(){
  22. System.out.println("测试用例3");
  23. }
  24. }

image.png

3.@BeforeEach、@AfterEach
  • @BeforeEach:表示被注解的方法应在当前类的每个@Test,@RepeatedTest,@ParameterizedTest或@TestFactory方法之前执行;
  • @AfterEach:表示被注解的方法应在当前类的每个@Test,@RepeatedTest,@ParameterizedTest或@TestFactory方法之后执行;
  1. public class JUnitTest {
  2. @BeforeAll
  3. static void beforeAll(){
  4. System.out.println("我是BeforeAll,我最开始执行。");
  5. }
  6. @AfterAll
  7. static void afterAll(){
  8. System.out.println("我是AfterAll,我最后执行。");
  9. }
  10. @BeforeEach
  11. void beforeEach(){
  12. System.out.println("我是BeforeEach,我在每个 @Test 前执行。");
  13. }
  14. @AfterEach
  15. void afterEach(){
  16. System.out.println("我是AfterEach,我在每个 @Test 后执行。");
  17. }
  18. @Test
  19. void test0(){
  20. System.out.println("测试用例1");
  21. }
  22. @Test
  23. void test1(){
  24. System.out.println("测试用例2");
  25. }
  26. @Test
  27. void test2(){
  28. System.out.println("测试用例3");
  29. }
  30. }

结果:

image.png

4.@Disabled

  @Disabled用于禁用测试类或测试方法,添加该注解的方法不会被测试。

  1. public class JUnitTest {
  2. @BeforeAll
  3. static void beforeAll(){
  4. System.out.println("我是BeforeAll,我最开始执行。");
  5. }
  6. @AfterAll
  7. static void afterAll(){
  8. System.out.println("我是AfterAll,我最后执行。");
  9. }
  10. @BeforeEach
  11. void beforeEach(){
  12. System.out.println("我是BeforeEach,我在每个 @Test 前执行。");
  13. }
  14. @AfterEach
  15. void afterEach(){
  16. System.out.println("我是AfterEach,我在每个 @Test 后执行。");
  17. }
  18. @Test
  19. @Disabled //忽略测试用例1
  20. void test0(){
  21. System.out.println("测试用例1");
  22. }
  23. @Test
  24. void test1(){
  25. System.out.println("测试用例2");
  26. }
  27. @Test
  28. void test2(){
  29. System.out.println("测试用例3");
  30. }
  31. }

结果:

image.png

(三)参数化测试

1.@ParameterizedTest + @ValueSource

  @ParameterizedTest的作用就是可以用不同的参数多次运行测试。但是必须声调用提供参数的来源(source)。

  @ValueSource它可以让你指定一个原生类型(String,int,long或double)的数组,并且只能为每次调用提供一个参数。

  1. public class JUnitTest {
  2. @BeforeAll
  3. static void beforeAll(){
  4. System.out.println("我是BeforeAll,我最开始执行。");
  5. }
  6. @AfterAll
  7. static void afterAll(){
  8. System.out.println("我是AfterAll,我最后执行。");
  9. }
  10. @BeforeEach
  11. void beforeEach(){
  12. System.out.println("我是BeforeEach,我在每个 Test 前执行。");
  13. }
  14. @AfterEach
  15. void afterEach(){
  16. System.out.println("我是AfterEach,我在每个 Test 后执行。");
  17. }
  18. @Test
  19. void test0(){
  20. System.out.println("测试用例1");
  21. }
  22. @Test
  23. void test1(){
  24. System.out.println("测试用例2");
  25. }
  26. @Test
  27. void test2(){
  28. System.out.println("测试用例3");
  29. }
  30. @ParameterizedTest
  31. @ValueSource(strings = {"小明","小红","小兰"})
  32. void paramTest(String name){
  33. System.out.println(name);
  34. }
  35. }

结果:

image.png

2.@ParameterizedTest + @CsvSource

  @CsvSource允许将参数列表表示为以逗号分隔的值(例如,字符串文字)。

  1. public class JUnitTest {
  2. @ParameterizedTest
  3. @CsvSource({"小明, 1","小红,2","小兰,3"})
  4. void csvSource(String name,int id){
  5. System.out.println(name + ":" + id);
  6. }
  7. }

结果:

image.png

@CsvSource使用'作为转义字符。

示例输入结果字符列表
@CsvSource({ "foo, bar" })"foo", "bar"
@CsvSource({ "foo, 'baz, qux'" })"foo", "baz, qux"
@CsvSource({ "foo, ''" })"foo", ""
@CsvSource({ "foo, " })"foo", null
3.@ParameterizedTest + @CsvFileSource

  @CsvFileSource让你使用classpath中的CSV文件。CSV文件中的每一行都会导致参数化测试的一次调用。

resources目录下创建csv文件:

test.csv:

  1. 小明, 1
  2. 小红, 2
  3. "小明, 小红", 3
  1. public class JUnitTest {
  2. @ParameterizedTest
  3. @CsvFileSource(resources = "/test.csv")
  4. void csvFile(String name,int id){
  5. System.out.println(name + ": " + id);
  6. }
  7. }

结果:

image.png

@CsvSource中使用的语法相反,@CsvFileSource使用双引号"作为转义字符。通过上面的代码就可以看出来。一个空的转义值""会产生一个空字符串, 一个完全为空的值被解释为null引用。

4.@ParameterizedTest + @MethodSource

  @MethodSource允许引用一个或多个测试类的工厂方法。

  1. public class JUnitTest {
  2. @ParameterizedTest
  3. @MethodSource("stringProvider") //指定方法
  4. void methodSource(int age,String name){
  5. System.out.println(age + ": " + name);
  6. }
  7. static Stream<Arguments> stringProvider() {
  8. return Stream.of(
  9. Arguments.arguments(12,"李四"),
  10. Arguments.arguments(18,"王五"),
  11. Arguments.arguments(20,"小红")
  12. );
  13. }
  14. }

  @MethodSource注解表示这个方法的参数来源于一个名为stringProvider的静态方法。stringProvider方法返回一个Stream<Arguments>类型的对象,其中每个Arguments对象包含了一组用于测试的参数。

image.png

(四)测试方法的执行顺序

1.@TestMethodOrder + @Order

  在 JUnit5 中,测试方法执行的顺序是不确定的或者是根据方法首字母来排序的。

  1. public class JUnitTest2 {
  2. @Test
  3. void C(){
  4. System.out.println("A");
  5. }
  6. @Test
  7. void B(){
  8. System.out.println("B");
  9. }
  10. @Test
  11. void A(){
  12. System.out.println("C");
  13. }
  14. }

结果:

image.png

  让执行顺序为 C、B、A:

  1. @TestMethodOrder(MethodOrderer.OrderAnnotation.class)
  2. public class JUnitTest2 {
  3. int a = 0;
  4. @Test
  5. @Order(1)
  6. void C(){
  7. a++;
  8. System.out.println(a);
  9. System.out.println("C");
  10. }
  11. @Test
  12. @Order(2)
  13. void B(){
  14. a++;
  15. System.out.println(a);
  16. System.out.println("B");
  17. }
  18. @Test
  19. @Order(3)
  20. void A(){
  21. a++;
  22. System.out.println(a);
  23. System.out.println("A");
  24. }
  25. }

  首先在类上添加@TestMethodOrder(MethodOrderer.OrderAnnotation.class),然后再为每个方法上添加@Order()注解,值越小越优先被执行。

(五)测试实例的生命周期

1.@TestInstance

  我添加一个成员变量,每次执行测试方法的时候都++一次。

  1. @TestMethodOrder(MethodOrderer.OrderAnnotation.class)
  2. @TestInstance(TestInstance.Lifecycle.PER_METHOD)
  3. public class JUnitTest2 {
  4. int a = 0;
  5. @Test
  6. @Order(1)
  7. void A(){
  8. a++;
  9. System.out.println("A方法:" + a);
  10. System.out.println("A");
  11. }
  12. @Test
  13. @Order(2)
  14. void B(){
  15. a++;
  16. System.out.println("B方法:" + a);
  17. System.out.println("B");
  18. }
  19. @Test
  20. @Order(3)
  21. void C(){
  22. a++;
  23. System.out.println("C方法:" + a);
  24. System.out.println("C");
  25. }
  26. }

结果:

image.png

  为了允许隔离执行单个的测试方法,JUnit在执行每个测试方法之前会创建每个测试类的新实例。如果想改变策略,就要用@TestInstance,在类上添加@TestInstance(TestInstance.Lifecycle.PER_CLASS)

  1. @TestMethodOrder(MethodOrderer.OrderAnnotation.class)
  2. @TestInstance(TestInstance.Lifecycle.PER_CLASS)
  3. public class JUnitTest2 {
  4. int a = 0;
  5. @Test
  6. @Order(1)
  7. void A(){
  8. a++;
  9. System.out.println("A方法:" + a);
  10. System.out.println("A");
  11. }
  12. @Test
  13. @Order(2)
  14. void B(){
  15. a++;
  16. System.out.println("B方法:" + a);
  17. System.out.println("B");
  18. }
  19. @Test
  20. @Order(3)
  21. void C(){
  22. a++;
  23. System.out.println("C方法:" + a);
  24. System.out.println("C");
  25. }
  26. }

结果:

image.png

Lifecycle.PER_CLASS表示只创建一个实例。不添加注解的时候,默认是Lifecycle.PER_METHOD

  当使用这种模式时,每个测试类将创建一个新的测试实例。因此,如果测试方法依赖于存储在实例变量中的状态,则可能需要在@BeforeEach@AfterEach方法中重置该状态(重置变量的值)。

(六)断言 Assertions

断言方法描述
assertEquals(expected, actual)检查两个值是否相等,如果不相等则抛出AssertionError
assertNotEquals(expected, actual)检查两个值是否不相等,如果相等则抛出AssertionError
assertTrue(condition)检查一个条件是否为真,如果为假则抛出AssertionError
assertFalse(condition)检查一个条件是否为假,如果为真则抛出AssertionError
assertNull(object)检查一个对象是否为null,如果不为null则抛出AssertionError
assertNotNull(object)检查一个对象是否不为null,如果为null则抛出AssertionError
assertSame(expected, actual)检查两个对象是否是同一个实例,如果不是则抛出AssertionError
assertNotSame(expected, actual)检查两个对象是否不是同一个实例,如果是则抛出AssertionError
assertArrayEquals(expected, actual)检查两个数组是否相等,如果不相等则抛出AssertionError
assertTimeout(duration, executable)检查一个可执行的代码块是否在指定的时间内完成,如果超时则抛出AssertionError
  1. public class JUnitTest3 {
  2. @Test
  3. void assertEqualsDemo(){
  4. int num = 10;
  5. Assertions.assertEquals(1,num,"不符合预期");
  6. }
  7. @Test
  8. void assertTrueDemo(){
  9. int num = 10;
  10. Assertions.assertTrue(num > 10,"不符合预期");
  11. }
  12. @Test
  13. void assertTimeoutDemo(){
  14. int num = 10;
  15. Assertions.assertTimeout(Duration.ofSeconds(3), new Executable() {
  16. @Override
  17. public void execute() throws Throwable {
  18. //代码块
  19. Thread.sleep(4000);
  20. }
  21. });
  22. }
  23. }
结果:

image.png

(七)测试套件

  测试套件是一组相关的测试,可以一起运行,以便更方便地组织和管理测试。使用套件要引入两个依赖:junit-platform-suite-apijunit-platform-suite-engine,具体的在文章开头。

  套件其实很好理解,就是使几个类同时进行测试。

1.@SelectClasses
  1. @Suite
  2. @SelectClasses(value = {JUnitTest.class,JUnitTest2.class})
  3. public class RunSuite {
  4. }

@Suite的作用是将一个类标记为JUnit平台上的测试套件。

@SelectClasses指定在JUnit平台上运行测试套件时要选择的类。

运行结果:

image.png

2.@SelectPackages

  可以选择类,那么也可以包。

  1. @Suite
  2. @SelectPackages(value = {"package1"})
  3. //可以选择多个包:@SelectPackages(value = {"package1","package2","package3"……})
  4. public class RunSuite {
  5. }

image.png

结果:

image.png

为什么只执行了JUnitTest这一个类?我的JUnitTest2呢?我们来看看它:

image.png

IDEA提示我们它的命名不符合规则,那这个规则是什么意思呢?

3.测试类命名规则
  • [A-Z[A-Za-z\d]*Test(s|Case)?:表示以大写字母开头,后面跟任意个字母或数字,最后以Test, Tests, TestCase结尾的字符串,例如MyTest, MyTests, MyTestCase等。
  • Test[A-Z[A-Za-z\d]*:表示以Test开头,后面跟一个大写字母,再后面跟任意个字母或数字的字符串,例如TestMyClass, TestMyMethod等。
  • IT(.*):表示以IT开头,后面跟任意个任意字符的字符串,例如ITMyClass, ITMyMethod等。
  • (.*)IT(Case)?:表示以任意个任意字符开头,后面跟IT或者ITCase的字符串,例如MyClassIT, MyMethodITCase等。

其实就是我们的类命名不规范导致框架识别不出来。改类名后:

image.png

image.png

最后感谢每一个认真阅读我文章的人,看着粉丝一路的上涨和关注,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走!

软件测试面试文档

我们学习必然是为了找到高薪的工作,下面这些面试题是来自阿里、腾讯、字节等一线互联网大厂最新的面试资料,并且有字节大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。
 

在这里插入图片描述

声明:本文内容由网友自发贡献,转载请注明出处:【wpsshop博客】
推荐阅读
相关标签
  

闽ICP备14008679号