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