在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
