是否可以使用Mockito验证测试的对象方法调用?

时间:2015-12-13 21:09:31

标签: java junit tdd mockito junit4

我'有一些业务逻辑类:

public class SomeService {
    public void doFirst() {}

    public void doSecond() {
        doFirst();
    }
}

并测试它:

public class SomeServiceTest {

    private SomeService service;

    @Before
    public void setUp() {
        service = new SomeService();
    }

    @Test
    public void doSecond_someCondition_shouldCallFirst() {

        // given
        ...

        // when
        service.doSecond();

        //then
        how to verify doFirst() was called? 
    }
}

如何验证doFirst()不是在模拟上调用,而是在实际服务中调用?

2 个答案:

答案 0 :(得分:8)

我想知道你为什么要测试你的测试方法调用什么方法。听起来很像白盒测试对我来说。

在我看来,你想要验证调用的结果而不是去那里的方法,因为这可能很容易改变(即重构时)。

因此,如果doSecond()的结果与doFirst()相同,则可以为doFirst()编写测试,并使用相同的测试(即断言集)来测试doSecond()。

但是如果你真的想测试,doSeirst()是否已被doSecond()调用,你可以将你的服务包装在间谍中,然后在间谍上调用验证:

//given
SomeService service = new SomeService();
SomeService spy = Mockito.spy(service);
//when
spy.doSecond();
//then
verify(spy).doFirst();

答案 1 :(得分:1)

听起来你想避免在测试中调用真正的doFirst吗?如果是这样,试试这个......

  //given
    boolean firstCalled = false;
    SomeService fakeService  = new SomeService {
          @Override 
          public void doFirst() {
             firstCalled = true;
          }
    }
 //when
 fakeService.doSecond();

  // then
  assertTrue(firstCalled);

这种测试/模拟技术被称为子类和覆盖'原因很明显。