Java 如何模拟从抽象类继承的受保护子类方法?
如何使用Mockito或PowerMock来模拟由子类实现但从抽象超类继承的受保护方法 换句话说,我想测试“doSomething”方法,同时模拟“doSomethingElse” 抽象超类Java 如何模拟从抽象类继承的受保护子类方法?,java,unit-testing,junit,mockito,powermock,Java,Unit Testing,Junit,Mockito,Powermock,如何使用Mockito或PowerMock来模拟由子类实现但从抽象超类继承的受保护方法 换句话说,我想测试“doSomething”方法,同时模拟“doSomethingElse” 抽象超类 public abstract class TypeA { public void doSomething() { // Calls for subclass behavior doSomethingElse(); } pr
public abstract class TypeA {
public void doSomething() {
// Calls for subclass behavior
doSomethingElse();
}
protected abstract String doSomethingElse();
}
子类实现
public class TypeB extends TypeA {
@Override
protected String doSomethingElse() {
return "this method needs to be mocked";
}
}
解决方案
public class TypeB extends TypeA {
@Override
protected String doSomethingElse() {
return "this method needs to be mocked";
}
}
这里给出的答案是正确的,如果所涉及的课程在同一个包中,那么这些答案将起作用
但是,如果涉及不同的包,一个选择是使用PowerMock。下面的例子对我很有用。当然,可能还有其他方法,这是一种有效的方法
import static org.junit.Assert.assertEquals;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.powermock.api.mockito.PowerMockito;
import org.powermock.core.classloader.annotations.PrepareForTest;
import org.powermock.modules.junit4.PowerMockRunner;
@RunWith(PowerMockRunner.class)
@PrepareForTest({ TypeB.class })
public class TestAbstract {
@Test
public void test_UsingPowerMock() throws Exception {
// Spy a subclass using PowerMock
TypeB b = PowerMockito.spy(new TypeB());
String expected = "some value for mock";
// Mock a method by its name using PowerMock again
PowerMockito.doReturn(expected).when(b, "doSomethingElse");
// Calls the
String actual = b.doSomething();
assertEquals(expected, actual);
}
}
注意:测试使用Java5、JUnit4.11、Mockito 1.9.0和PowerMock 1.4.12完成 您可以通过以下方式使用mockito测试abtract类
TypeA typA = Mockito.mock(TypeA.class, Mockito.CALLS_REAL_METHODS);
when(typA.doSomethingElse()).thenReturn("doSomethingElse");
Assert.assertSomething(typeA.doSomething());
我建议使用Mockito:
// Create a new Mock
final TypeA a = Mockito.mock(TypeA.class, Mockito.CALLS_REAL_METHODS);
// Call the method
a.doSomething();
// Now verify that our mocked class' method was called
Mockito.verify(a, Mockito.times(1)).doSomethingElse();
在模拟抽象方法时,可以使用
Mockito.CALLS\u REAL\u方法
。这将调用类的原始方法,您可以自己模拟所有抽象方法
TypeA typeA = mock(TypeA.class, Mockito.CALLS_REAL_METHODS);
when(typeA.doSomethingElse()).thenReturn("Hello");
typeA.doSomething();
或者直接在TypeB上使用spy进行测试:
TypeB typeB = spy(new TypeB());
when(typeB.doSomethingElse()).thenReturn("Hello");
typeB.doSomething();
要模拟在抽象类中返回void的方法,我们可以使用:
MyAbstractClass abs = Mockito.mock(MyAbstractClass.class);
Mockito.doNothing().when(abs).myMethod(arg1,arg2....));
我们可以根据需要用Mockito.anyString()等替换参数。第一个选项失败,因为TypeA是抽象的。选项2与
spy()
配合使用,但当我尝试将其应用于生产代码时,它会给我类型中的方法不可见。试图找出原因。实际上你的测试是正确的,但我有不同包中的类,这就是为什么我不能早些做的原因。方法在受保护时在其他包中不可见。第一个选项也有效吗?我已经用Mockito 1.10.19和最新的beta版本测试了这两个版本,它们都按预期工作。确实有效。忘了告诉你我的Mockito版本很古老,1.9.0,因为Java5。它可以工作,但我刚刚意识到我的生产代码中有不同包中的类。此时受保护的方法不可见。这使它成为一个问题。一个可能的解决方案是使用PowerMock,它应该能够验证私有方法afaik。但从未使用过它。这种方法让我对Java 8、JUnit 4.12和PowerMock 1.6.4产生了NullPointerException
。@Mack你解决了吗?@ceyun我很抱歉,但我已经很久不记得我是否解决了它,或者我是如何解决的了。也许是对我的评论投了赞成票的人之一?