使用PowerMockito.whenNew()不会被模拟并调用原始方法

use*_*446 82 java junit mockito powermock

我有一个类似下面的代码:

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;
  }
}
Run Code Online (Sandbox Code Playgroud)

现在我正在写一个测试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
Run Code Online (Sandbox Code Playgroud)

我曾经@RunWith(PowerMockRunner.class)@PrepareForTest({MyQueryClass.class})我的测试类的开始.

但是,当我调试测试,它仍然是调用真正的方法getNextId()中的MyQueryClass类.

我在这里错过了什么?任何人都可以提供帮助,因为我是Mockito和PowerMockito的新手.

Tru*_*Dub 175

您需要将构造函数调用的类放入@PrepareForTest注释而不是正在构造的类中 - 请参阅Mock构造新对象.

在你的情况下:

@PrepareForTest(MyQueryClass.class)

@PrepareForTest(A.class)

更一般:

@PrepareForTest(NewInstanceClass.class)

@PrepareForTest(ClassThatCreatesTheNewInstance.class)

  • 我有同样的问题,但这个解决方案不适合我 (3认同)
  • 多谢。在 @PrepareForTest 中包含当前类 Eg A 后,它现在可以工作了。 (2认同)
  • 我也花了一些时间.谢谢@TrueDub.因为参考已经过时了.我只是更新它.https://github.com/jayway/powermock/wiki/MockConstructor它说:在测试用例的类级别使用@PrepareForTest(ClassThatCreatesTheNewInstance.class)注释. (2认同)
  • 如果您使用eclemma进行代码覆盖,则此解决方案将不起作用。将要测试的类添加到@PrepareForTest将导致该类的覆盖率为0% (2认同)
  • 解决方案将起作用 - 测试正确执行.显然,eclema无法应对PowerMockito.代码覆盖率不是此问题的一部分. (2认同)

ACV*_*ACV 6

正如@TrueDub 在他接受的回复中提到的,您需要将调用构造函数的类添加到@PrepareForTest.

但是,如果您这样做,则 eclemma 和 Sonar 报告的该类的覆盖率对于该类将为零

Powermockito 维基

我们将用 ByteBuddy (#727) 替换 Javassist,它应该有助于解决这个老问题。但是现在没有办法将 PowerMock 与 JaCoCo On-the-fly 仪器一起使用。并且没有解决方法可以在 IDE 中获得代码覆盖率。

所以这里的解决方案是重构实际代码以使用静态工厂,该工厂将返回该类的实例,然后静态模拟它。