在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
方法专门测试异常:
@Testvoid 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开始引入了函数式编程,所有单方法接口都可以简写如下:
@Testvoid testNegative() { assertThrows(IllegalArgumentException.class, () -> { Factorial.fact(-1); }); }