我的代码如下所示:
Class A {
public boolean myMethod(someargs) {
MyQueryClass query = new MyQueryClass();
Long id = query.getNextId();
// some more code
}
}
Class MyQueryClass {
....
public Long getNextId() {
//lot of DB code, execute some DB query
return id;
}
}
现在我正在为A.myMethod(someargs)
编写测试。我想跳过真正的方法query.getNextId()
,而是返回一个存根值。基本上,我想模仿MyQueryClass
。
所以在我的测试用例中,我使用过:
MyQueryClass query = PowerMockito.mock(MyQueryClass.class);
PowerMockito.whenNew(MyQueryClass.class).withNoArguments().thenReturn(query);
when(query.getNextId()).thenReturn(1000000L);
boolean b = A.getInstance().myMethod(args);
//asserts
我在测试类的开头使用了@RunWith(PowerMockRunner.class)
和@PrepareForTest({MyQueryClass.class})
。
但是当我调试测试时,它仍在调用getNextId()
类的实际方法MyQueryClass
。
我在这里缺少什么?任何人都可以帮忙,因为我是Mockito和PowerMockito的新手。
答案 0 :(得分:154)
您需要将构造函数调用的类放入@PrepareForTest
注释而不是正在构造的类中 - 请参阅Mock construction of new objects。
在你的情况下:
✗@PrepareForTest(MyQueryClass.class)
✓@PrepareForTest(A.class)
更一般:
✗@PrepareForTest(NewInstanceClass.class)
✓@PrepareForTest(ClassThatCreatesTheNewInstance.class)
答案 1 :(得分:1)
正如@TrueDub在其接受的答复中提到的那样,您需要将调用构造函数的类添加到@PrepareForTest
中。
但是,如果执行此操作,则eclemma和Sonar报告的该课程的覆盖率对该课程而言为零
我们将用ByteBuddy(#727)替换Javassist,它应该 帮助解决这个旧问题。但是现在没有办法使用 带有JaCoCo动态仪器的PowerMock。而且没有解决方法 在IDE中获得代码覆盖率。
因此,这里的解决方案是将实际代码重构为使用静态工厂,该工厂将返回该类的实例,然后对其进行静态模拟。
答案 2 :(得分:0)
也许您可以简单地使用
Mockito.doReturn(value).when(xxx)