共计 1174 个字符,预计需要花费 3 分钟才能阅读完成。
在 Java 程序中,异常处理是非常重要的。
我们自己编写的方法,也经常抛出各种异常。对于可能抛出的异常进行测试,本身就是测试的重要环节。
因此,在编写 JUnit 测试的时候,除了正常的输入输出,我们还要特别针对可能导致异常的情况进行测试。
我们仍然用 Factorial
举例:
public class Factorial {public static long fact(long n) {if (n < 0) {throw new IllegalArgumentException();}
long r = 1;
for (long i = 1; i <= n; i++) {r = r * i;}
return r;
}
}
在方法入口,我们增加了对参数 n
的检查,如果为负数,则直接抛出IllegalArgumentException
。
现在,我们希望对异常进行测试。在 JUnit 测试中,我们可以编写一个 @Test
方法专门测试异常:
@Test
void testNegative() {assertThrows(IllegalArgumentException.class, new Executable() {@Override
public void execute() throws Throwable {Factorial.fact(-1);
}
});
}
JUnit 提供 assertThrows()
来期望捕获一个指定的异常。第二个参数 Executable
封装了我们要执行的会产生异常的代码。当我们执行 Factorial.fact(-1)
时,必定抛出 IllegalArgumentException
。assertThrows()
在捕获到指定异常时表示通过测试,未捕获到异常,或者捕获到的异常类型不对,均表示测试失败。
有些童鞋会觉得编写一个 Executable
的匿名类太繁琐了。实际上,Java 8 开始引入了函数式编程,所有单方法接口都可以简写如下:
@Test
void testNegative() {assertThrows(IllegalArgumentException.class, () -> {Factorial.fact(-1);
});
}
上述奇怪的 ->
语法就是函数式接口的实现代码,我们会在后面详细介绍。现在,我们只需要通过这种固定的代码编写能抛出异常的语句即可。
练习
观察 Factorial.fact()
方法,注意到由于 long
型整数有范围限制,当我们传入参数 21
时,得到的结果是 -4249290049419214848
,而不是期望的51090942171709440000
,因此,当传入参数大于20
时,Factorial.fact()
方法应当抛出ArithmeticException
。请编写测试并修改实现代码,确保测试通过。
下载练习
小结
测试异常可以使用assertThrows()
,期待捕获到指定类型的异常;
对可能发生的每种类型的异常都必须进行测试。