在Mockito中检测到未完成的存根
运行测试时出现以下异常。我正在使用Mockito进行嘲笑。Mockito库提到的提示无济于事。
org.mockito.exceptions.misusing.UnfinishedStubbingException: Unfinished stubbing detected here:
-> at com.a.b.DomainTestFactory.myTest(DomainTestFactory.java:355)
E.g. thenReturn() may be missing.
Examples of correct stubbing:
when(mock.isOk()).thenReturn(true);
when(mock.isOk()).thenThrow(exception);
doThrow(exception).when(mock).someVoidMethod();
Hints:
1. missing thenReturn()
2. you are trying to stub a final method, you naughty developer!
at a.b.DomainTestFactory.myTest(DomainTestFactory.java:276)
..........
来自的测试代码DomainTestFactory
。当我运行以下测试时,我看到了异常。
@Testpublic myTest(){
MyMainModel mainModel = Mockito.mock(MyMainModel.class);
Mockito.when(mainModel.getList()).thenReturn(getSomeList()); // Line 355
}
private List<SomeModel> getSomeList() {
SomeModel model = Mockito.mock(SomeModel.class);
Mockito.when(model.getName()).thenReturn("SomeName"); // Line 276
Mockito.when(model.getAddress()).thenReturn("Address");
return Arrays.asList(model);
}
public class SomeModel extends SomeInputModel{
protected String address;
protected List<SomeClass> properties;
public SomeModel() {
this.Properties = new java.util.ArrayList<SomeClass>();
}
public String getAddress() {
return this.address;
}
}
public class SomeInputModel{
public NetworkInputModel() {
this.Properties = new java.util.ArrayList<SomeClass>();
}
protected String Name;
protected List<SomeClass> properties;
public String getName() {
return this.Name;
}
public void setName(String value) {
this.Name = value;
}
}
回答:
您是在嘲笑内部嵌套嘲笑。getSomeList()
在完成对的模拟之前,您正在呼叫,它会进行一些模拟MyMainModel
。执行此操作时,Mockito不喜欢它。
更换
@Testpublic myTest(){
MyMainModel mainModel = Mockito.mock(MyMainModel.class);
Mockito.when(mainModel.getList()).thenReturn(getSomeList()); --> Line 355
}
与
@Testpublic myTest(){
MyMainModel mainModel = Mockito.mock(MyMainModel.class);
List<SomeModel> someModelList = getSomeList();
Mockito.when(mainModel.getList()).thenReturn(someModelList);
}
要了解为什么这会导致问题,您需要稍微了解Mockito的工作方式,并且还需要知道Java中表达式和语句按什么顺序求值。
Mockito无法读取您的源代码,因此为了弄清楚您要执行的操作,它在很大程度上取决于静态状态。在模拟对象上调用方法时,Mockito会在内部调用列表中记录调用的详细信息。该when
方法从列表中读取这些调用中的最后一个,并将此调用记录在OngoingStubbing
它返回的对象中。
线
Mockito.when(mainModel.getList()).thenReturn(someModelList);
导致与Mockito的以下交互:
- 模拟方法
mainModel.getList()
被称为 - 静态方法
when
被称为 - 在方法返回
thenReturn
的OngoingStubbing
对象上调用when
方法。
thenReturn
然后,该方法可以指示通过该OngoingStubbing
方法接收到的模拟对象,以处理对该方法的任何合适的调用getList
以返回someModelList
。
实际上,由于Mockito看不到您的代码,因此您还可以按以下方式编写模拟:
mainModel.getList();Mockito.when((List<SomeModel>)null).thenReturn(someModelList);
这种样式的读起来不太清晰,特别是因为在这种情况下null
必须强制转换,但是它与Mockito产生相同的交互序列,并且将获得与上面的行相同的结果。
但是,线
Mockito.when(mainModel.getList()).thenReturn(getSomeList());
导致与Mockito的以下交互:
- 模拟方法
mainModel.getList()
被称为 - 静态方法
when
被称为 mock
在SomeModel
中创建了一个新的getSomeList()
,- 模拟方法
model.getName()
被称为
此时,Mockito感到困惑。它以为您在嘲笑mainModel.getList()
,但是现在您要告诉它您要模拟该model.getName()
方法。对于Mockito来说,您似乎正在执行以下操作:
when(mainModel.getList());// ...
when(model.getName()).thenReturn(...);
这看起来很愚蠢,Mockito
因为无法确定您在做什么mainModel.getList()
。
请注意,我们尚未进行thenReturn
方法调用,因为JVM需要先评估此方法的参数,然后才能调用该方法。在这种情况下,这意味着调用该getSomeList()
方法。
通常,像Mockito一样,依靠静态是一个错误的设计决策,因为它可能导致违反“最小惊讶原则”的情况。但是,Mockito的设计确实可以进行清晰而富有表现力的嘲讽,即使有时会令人惊讶。
最后,最新版本的Mockito在上面的错误消息中增加了一行。此额外的行表示您可能正处于与此问题相同的情况:
3:在完成“ thenReturn”指令之前,您正在测试另一个模拟的行为
以上是 在Mockito中检测到未完成的存根 的全部内容, 来源链接: utcz.com/qa/421660.html