18.单元测试

单元测试

单元测试是针对最小的功能单元编写测试代码
Java程序最小的功能单元是方法,对Java程序进行单元测试就是针对单个Java方法的测试
测试驱动开发TDD是一种根据功能实现代码的开发模式

JUnit

Java平台最常用的测试框架JUnit是一个开源的Java语言的单元测试框架,专门针对Java设计,使用最广泛
编写了一个xxx.java文件后,我们想对其进行测试,需要编写一个对应的xxxTest.java文件,以Test为后缀是一个惯例,并分别将其放入src和test目录中
一个JUnit测试包含若干@Test方法,并使用Assertions进行断言,核心测试方法testFact()要加上@Test注解,JUnit会把带有@Test的方法识别为测试方法。Assertion定义的用于测试的断言方法包括:

  • assertEquals(expected, actual):期待actual结果等于expected
  • assertTrue(): 期待结果为true
  • assertFalse(): 期待结果为false
  • assertNotNull(): 期待结果为非null
  • assertArrayEquals(): 期待结果为数组并与期望数组每个元素的值均相等

如果测试结果与预期不符,assertEquals()会抛出异常,我们就会得到一个测试失败的结果
单元测试可以确保单个方法按照正确预期运行,如果修改了某个方法的代码,只需确保其对应的单元测试通过,即可认为改动正确。此外,测试代码本身就可以作为示例代码,用来演示如何调用该方法。
使用断言(Assertion)来测试期望结果,可以方便地组织和运行测试,并方便地查看测试结果。此外,JUnit既可以直接在IDE中运行,也可以方便地集成到Maven这些自动化工具中运行。

单元测试需遵循规范:

  • 单元测试代码本身必须非常简单,能一下看明白,决不能再为测试代码编写测试
  • 每个单元测试应当互相独立,不依赖运行的顺序
  • 测试时不但要覆盖常用测试用例,还要特别注意测试边界条件,例如输入为0,null,空字符串""等情况

Fixture

在一个单元测试中,经常编写多个@Test方法,来分组、分类对目标代码进行测试。在测试的时候,经常遇到一个对象需要初始化,测试完可能还需要清理。JUnit提供了编写测试前准备、测试后清理的固定代码,我们称之为Fixture
编写Fixture是指针对每个@Test方法,编写@BeforeEach方法用于初始化测试资源,编写@AfterEach用于清理测试资源;必要时,可以编写@BeforeAll和@AfterAll,使用静态变量来初始化耗时较长的资源,并且在所有@Test方法的运行前后仅执行一次。Test结构如下

invokeBeforeAll(CalculatorTest.class);
for (Method testMethod : findTestMethods(CalculatorTest.class)) {
    var test = new CalculatorTest(); // 创建Test实例
    invokeBeforeEach(test);
        invokeTestMethod(test, testMethod);
    invokeAfterEach(test);
}
invokeAfterAll(CalculatorTest.class);

对于实例变量,在@BeforeEach中初始化,在@AfterEach中清理,它们在各个@Test方法中互不影响,因为是不同的实例;对于静态变量,在@BeforeAll中初始化,在@AfterAll中清理,它们在各个@Test方法中均是唯一实例,会影响各个@Test方法。
大多数情况下,使用@BeforeEach和@AfterEach足够。只有某些测试资源初始化耗费时间太长,以至于我们不得不尽量“复用”时才会用到@BeforeAll和@AfterAll。
最后,注意到每次运行一个@Test方法前,JUnit首先创建一个XxxTest实例,因此,每个@Test方法内部的成员变量都是独立的,不能也无法把成员变量的状态从一个@Test方法带到另一个@Test方法

异常测试

在编写JUnit测试的时候,除了正常的输入输出,我们还要特别针对可能导致异常的情况进行测试。测试异常可以使用assertThrows(),期待捕获到指定类型的异常;对可能发生的每种类型的异常都必须进行测试。
JUnit提供assertThrows()来期望捕获一个指定的异常。第二个参数Executable封装了我们要执行的会产生异常的代码。当我们执行测试代码并抛出IllegalArgumentException时,assertThrows()在捕获到指定异常时表示通过测试,未捕获到异常,或者捕获到的异常类型不对,均表示测试失败。

条件测试

条件测试是根据某些注解在运行期让JUnit自动忽略某些测试。JUnit根据不同的条件注解,决定是否运行当前的@Test方法。例如
@Disabled:JUnit仍然识别出这是个测试方法,只是暂时不运行
@EnableOnOs:在某个系统运行
@DisabledOnOs(OS.WINDOWS):不在Windows平台执行的测试
@DisabledOnJre(JRE.JAVA_8):只能在Java 9或更高版本执行的测试
@EnabledIfSystemProperty(named = “os.arch”, matches = “.64.”):只能在64位操作系统上执行的测试
@EnabledIfEnvironmentVariable(named = “DEBUG”, matches = “true”):需要传入环境变量DEBUG=true才能执行的测试

参数化测试

如果待测试的输入和输出是一组数据: 可以把测试数据组织起来 用不同的测试数据调用相同的测试方法
参数化测试下,一个测试方法需要接收至少一个参数,然后传入一组参数反复运行。
JUnit提供了一个@ParameterizedTest注解,用来进行参数化测试。

参数如何传递?既可以在测试代码中写死,也可以通过@CsvFileSource放到外部的CSV文件中。

  • 通过@MethodSource注解,允许我们编写一个同名的静态方法来提供测试参数
@ParameterizedTest
@MethodSource
void testCapitalize(String input, String result) {
    assertEquals(result, StringUtils.capitalize(input));
}

static List<Arguments> testCapitalize() {
    return List.of( // arguments:
            Arguments.arguments("abc", "Abc"), //
            Arguments.arguments("APPLE", "Apple"), //
            Arguments.arguments("gooD", "Good"));
}
  • 使用@CsvSource,它的每一个字符串表示一行,一行包含的若干参数用,分隔
@ParameterizedTest
@CsvSource({ "abc, Abc", "APPLE, Apple", "gooD, Good" })
void testCapitalize(String input, String result) {
    assertEquals(result, StringUtils.capitalize(input));
}
  • 可以把测试数据提到一个独立的CSV文件中,然后标注上@CsvFileSource,JUnit只在classpath中查找指定的CSV文件,因此,test-capitalize.csv这个文件要放到test目录下
@ParameterizedTest
@CsvFileSource(resources = { "/test-capitalize.csv" })
void testCapitalizeUsingCsvFile(String input, String result) {
    assertEquals(result, StringUtils.capitalize(input));
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值