【发布时间】:2015-04-21 13:16:01
【问题描述】:
我有一个返回异常的函数,我正在 Junit4 中为它编写一个单元测试用例。问题是,reflect.invoke 总是将异常包装在 InvocationTargetException 中,因此无法使用 ExpectedException 类检查异常。
public class Hello {
private void printHello(String msg) {
if ("hello".equals(msg)) {
System.out.println("Hello");
}
else throw new HeaderException();
}
}
测试:
@Rule
public ExpectedException expectedEx = ExpectedException.none();
@Test
public void testPrint() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
expectedEx.expect(HeaderException.class);
Method method = Hello.class.getDeclaredMethod("printHello", String.class);
method.setAccessible(true);
method.invoke(Hello.class,"random");
}
输出:
java.lang.AssertionError:
Expected: an instance of com.locationguru.CSF.exception.HeaderException
got: <java.lang.reflect.InvocationTargetException>
<Click to see difference>
at org.junit.Assert.assertThat(Assert.java:780)
at org.junit.Assert.assertThat(Assert.java:738)
at org.junit.rules.ExpectedException$ExpectedExceptionStatement.evaluate(ExpectedException.java:114)
at org.junit.rules.RunRules.evaluate(RunRules.java:18)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:263)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:68)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:47)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:231)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:60)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:229)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:50)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:222)
at org.junit.runners.ParentRunner.run(ParentRunner.java:300)
at org.junit.runner.JUnitCore.run(JUnitCore.java:157)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:78)
at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:212)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:68)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:140)
【问题讨论】:
-
我强烈反对测试这样的私有方法。如果您需要它是可见的,但不是太可见,考虑将其设为包私有。
-
起初我以为
expectedEx.expectCause(instanceOf(HeaderException.class));会这样做,但不幸的是,HeaderException 没有存储在cause中,而是存储在target中。我建议您编写自己的 Matcher 或在网上搜索通用的 ExceptionMatcher。
标签: java reflection junit4 invoke