Mockito

作者: lv_shun | 来源:发表于2020-09-28 00:23 被阅读0次

在单元测试中,为了隔离外部资源,使用mock构造一个虚拟对象,保证测试的目标函数能正常运行。有很多Mock方式,Mockito是其中比较通用的。

基本mock

通常有两种方式引入mockito来进行mock:

  • 注解
@RunWith(MockitoJUnitRunner.class)
public class TestMockito {

    @Mock
    private AbstractConsumer abstractConsumer;

    @Test
    public void testMock() {
        abstractConsumer.execute("");
    }
}
  • 代码方式
public class TestMockito2 {

    @Mock
    private AbstractConsumer abstractConsumer;

    @Before
    public void init() {
        MockitoAnnotations.initMocks(this);
    }

    @Test
    public void test() {
        abstractConsumer.execute("");
    }
}
  • 当Mock对象调用方法返回对象也需要Mock时,可以使用@Mock注解中answer变量
public class DeepMockTest {
    //通过answer方式可以完成深度mock
    @Mock(answer = Answers.RETURNS_DEEP_STUBS)
    public IndexController indexController;
    @Before
    public void init() {
        MockitoAnnotations.initMocks(this);
    }
    @Test
    public void deepTest() {
        indexController.helloStudent("").toString();
    }
}

stub

在单元测试中,如果想调用某些方法时,想预先设定返回结果,可以通过stub来实现。

  • 调用mock对象中有返回值时,录制预期结果、调用、验证调用
@RunWith(MockitoJUnitRunner.class)
public class StubbingTest {

    public List list;

    @Before
    public void init() {
        list = mock(List.class);
    }
    
    /**
     * 对调用有返回值的函数调用stub预期结果
     */
    @Test
    public void test() {
        //stub
        when(list.get(0)).thenReturn("first");
        //验证
        assertThat(list.get(0),equalTo("first"));
        //stub anyInt , then throw exception
        when(list.get(anyInt())).thenThrow(new RuntimeException());
        try{
            list.get(0);
            fail();
        } catch (Exception e) {
            assertThat(e, instanceOf(RuntimeException.class));
        }
    }

    @After
    public void destory() {
        reset(list);
    }
}
  • 调用mock对象中无返回值的方法时,如何验证是否调用、验证抛出异常
@Test
public void noReturnStubTest() {
    //什么都不做
    doNothing().when(list).clear();
    //调用
    list.clear();
    //验证是否执行过一次
    verify(list,times(1)).clear();
    //调用无返回值clear时抛出异常
    doThrow(RuntimeException.class).when(list).clear();
    try {
        list.clear();
        fail();
    } catch (Exception e) {
        assertThat(e, instanceOf(RuntimeException.class));
    }
    //验证是否调用过
    verify(list,times(2)).clear();
}
  • 调用mock对象有返回值的函数,调用多次时,可以预设置每一次返回的结果
/**
 * stub相同方法的不同调用次数时的返回值
 */
@Test
public void MultiCallReturnMethodStubTest() {
    when(list.size()).thenReturn(1,2,3,4);
    assertThat(list.size(),equalTo(1));
    assertThat(list.size(),equalTo(2));
    assertThat(list.size(),equalTo(3));
    assertThat(list.size(),equalTo(4));
}
  • 调用mock对象由参数的方法时,需要根据参数来返回不同的返回值,需要thenanswer来完成
@Test
public void MultiCallArgMethodReturnStubTest() {
    when(list.get(anyInt())).thenAnswer(invocation -> {
        Integer argument = invocation.getArgument(0);
        return String.valueOf(argument * 10);
    });

    assertThat(list.get(1),equalTo("10"));
    assertThat(list.get(99),equalTo("990"));
}
  • 调用mock对象时可以stub预期结果,可以调用到mock的源对象本身方法,通过thenCallRealMethod
@Test
public void callRealMethodTest() {
    //mock是通过cglib生成代理对象
    Service service = mock(Service.class);
    //调用mock方法时stub返回值
    when(service.callMock()).thenReturn("mock123");
    assertThat(service.callMock(),equalTo("mock123"));
    //调用mock对象本身的方法
    when(service.callReal()).thenCallRealMethod();
    assertThat(service.callReal(),equalTo(10));
}

spy

spy也是对目标对象进行mock,但是只有设置了stub的方法才会mock,其他方法直接调用目标对象本身的方法,有两种方式代码方式和annotation方式。

  • 代码方式
@RunWith(MockitoJUnitRunner.class)
public class SpyTest {

    @Test
    public void spyTest() {
        //创建实例对象
        List<Integer> realList = new ArrayList<>();
        //spy包装实例对象
        List<Integer> list = spy(realList);
        //对spy对象设置stub
        when(list.isEmpty()).thenReturn(true);
        when(list.size()).thenReturn(10);
        //调用realList中的真实方法
        list.add(1);
        list.add(2);
        assertThat(list.get(0),equalTo(1));
        assertThat(list.get(1),equalTo(2));
        assertThat(list.size(),equalTo(10));
        assertThat(list.isEmpty(),equalTo(true));
    }
}
  • 注解方式
public class SpyAnnotationTest {
    @Spy
    public List<Integer> list = new ArrayList<>();

    @Before
    public void init() {
        MockitoAnnotations.initMocks(this);
    }
    @Test
    public void test() {
        when(list.size()).thenReturn(1);
        when(list.isEmpty()).thenReturn(true);
        list.add(1);
        list.add(2);
        assertThat(list.get(0),equalTo(1));
        assertThat(list.get(1),equalTo(2));
        assertThat(list.isEmpty(),equalTo(true));
        assertThat(list.size(),equalTo(1));
    }
}

相关文章

网友评论

      本文标题:Mockito

      本文链接:https://www.haomeiwen.com/subject/gouruktx.html