03、瞻前顾后
在一个测试用例中,可能要对多个方法进行测试。在测试之前呢,需要准备一些条件,比如说创建对象;在测试完成后呢,需要把这些对象销毁掉以释放资源。如果在多个测试方法中重复这些样板代码又会显得非常啰嗦。
这时候,该怎么办呢?
我为你提供了 setUp() 和 tearDown(),作为一个文化人,我称之为“瞻前顾后”。来看要测试的代码。
public class Calculator { public int sub(int a, int b) { return a - b; } public int add(int a, int b) { return a + b; } }
新建测试用例的时候记得勾选setUp 和 tearDown。
生成后的代码如下所示。
class CalculatorTest { Calculator calculator; @BeforeEach void setUp() { calculator = new Calculator(); } @AfterEach void tearDown() { calculator = null; } @Test void sub() { assertEquals(0,calculator.sub(1,1)); } @Test void add() { assertEquals(2,calculator.add(1,1)); } }
@BeforeEach 的 setUp() 方法会在运行每个 @Test 方法之前运行;@AfterEach 的 tearDown() 方法会在运行每个 @Test 方法之后运行。
与之对应的还有 @BeforeAll 和 @AfterAll,与 @BeforeEach 和 @AfterEach 不同的是,All 通常用来初始化和销毁静态变量。
public class DatabaseTest {
static Database db;
@BeforeAll
public static void init() {
db = createDb(...);
}
@AfterAll
public static void drop() {
...
}
}
03、异常测试
对于 Java 程序来说,异常处理也非常的重要。对于可能抛出的异常进行测试,本身也是测试的一个重要环节。
还拿之前的 Factorial 类来进行说明。在 fact() 方法的一开始,对参数 n 进行了校验,如果小于 0,则抛出 IllegalArgumentException 异常。
public class Factorial {
public static long fact(long n) {
if (n < 0) {
throw new IllegalArgumentException("参数不能小于 0");
}
long r = 1;
for (long i = 1; i <= n; i++) {
r = r * i;
}
return r;
}
}
在 FactorialTest 中追加一个测试方法 factIllegalArgument()。
@Test
void factIllegalArgument() {
assertThrows(IllegalArgumentException.class, new Executable() {
@Override
public void execute() throws Throwable {
Factorial.fact(-2);
}
});
}
我为你提供了一个 assertThrows() 的方法,第一个参数是异常的类型,第二个参数 Executable,可以封装产生异常的代码。如果觉得匿名内部类写起来比较复杂的话,可以使用 Lambda 表达式。
@Test
void factIllegalArgumentLambda() {
assertThrows(IllegalArgumentException.class, () -> {
Factorial.fact(-2);
});
}
04、忽略测试
有时候,由于某些原因,某些方法产生了 bug,需要一段时间去修复,在修复之前,该方法对应的测试用例一直是以失败告终的,为了避免这种情况,我为你提供了 @Disabled 注解。
class DisabledTestsDemo { @Disabled("该测试用例不再执行,直到编号为 43 的 bug 修复掉") @Test void testWillBeSkipped() { } @Test void testWillBeExecuted() { } }
@Disabled 注解也可以不需要说明,但我建议你还是提供一下,简单地说明一下为什么这个测试方法要忽略。在上例中,如果团队的其他成员看到说明就会明白,当编号 43 的 bug 修复后,该测试方法会重新启用的。即便是为了提醒自己,也很有必要,因为时间长了你可能自己就忘了,当初是为什么要忽略这个测试方法的。
05、条件测试
有时候,你可能需要在某些条件下运行测试方法,有些条件下不运行测试方法。针对这场使用场景,我为你提供了条件测试。
1)不同的操作系统,可能需要不同的测试用例,比如说 Linux 和 Windows 的路径名是不一样的,通过 @EnabledOnOs 注解就可以针对不同的操作系统启用不同的测试用例。
@Test @EnabledOnOs(MAC) void onlyOnMacOs() { // ... } @TestOnMac void testOnMac() { // ... } @Test @EnabledOnOs({ LINUX, MAC }) void onLinuxOrMac() { // ... } @Test @DisabledOnOs(WINDOWS) void notOnWindows() { // ... }
2)不同的 Java 运行环境,可能也需要不同的测试用例。@EnabledOnJre 和 @EnabledForJreRange 注解就可以满足这个需求。
@Test @EnabledOnJre(JAVA_8) void onlyOnJava8() { // ... } @Test @EnabledOnJre({ JAVA_9, JAVA_10 }) void onJava9Or10() { // ... } @Test @EnabledForJreRange(min = JAVA_9, max = JAVA_11) void fromJava9to11() { // ... }
06、尾声
最后,给你说三句心里话吧。在编写单元测试的时候,你最好这样做:
1)单元测试的代码本身必须非常名单明了,能一下看明白,决不能再为测试代码编写测试代码。
2)每个单元测试应该互相独立,不依赖运行时的顺序。
3)测试时要特别注意边界条件,比如说 0,null,空字符串"" 等情况。
希望我能尽早的替你发现代码中的 bug,毕竟越早的发现,造成的损失就会越小。see you!
推荐阅读:
鹅厂大佬掏心掏肺的分享:我的编程能力从什么时候开始突飞猛进?
Github 疯传!史上最强!BAT 大佬「LeetCode刷题手册」电子书开放下载了!
如果觉得有帮助的话,请不要吝啬一键三连哦,我知道,你们都是有素质的 CSDN 好公民,2021 年,让我们一起加油!