Mockito 是一个强大的用于 Java 开发的模拟测试框架, 经过 Mockito 咱们能够建立和配置 Mock 对象, 进而简化有外部依赖的类的测试.
使用 Mockito 的大体流程以下:segmentfault
建立外部依赖的 Mock 对象, 而后将此 Mock 对象注入到测试类中.框架
执行测试代码.单元测试
校验测试代码是否执行正确.测试
咱们已经知道了 Mockito 主要的功能就是建立 Mock 对象, 那么什么是 Mock 对象呢? 对 Mock 对象不是很了解的朋友, 能够参考这篇文章.
如今咱们对 Mock 对象有了必定的了解了, 那么天然就会有人问了, 为何要使用 Mock 对象? 使用它有什么好处呢?
下面咱们以一个简单的例子来展现一下 Mock 对象到底有什么用.
假设咱们正在编写一个银行的服务 BankService, 这个服务的依赖关系以下:spa
当咱们须要测试 BankService 服务时, 该真么办呢?
一种方法是构建真实的 BankDao, DB, AccountService 和 AuthService 实例, 而后注入到 BankService 中.
不用我说, 读者们也确定明白, 这是一种既笨重又繁琐的方法, 彻底不符合单元测试的精神. 那么还有一种更加优雅的方法吗? 天然是有的, 那就是咱们今天的主角 Mock Object. 下面来看一下使用 Mock 对象后的框架图:code
咱们看到, BankDao, AccountService 和 AuthService 都被咱们使用了虚拟的对象(Mock 对象) 来替换了, 所以咱们就能够对 BankService 进行测试, 而不须要关注它的复杂的依赖了.对象
为了简洁期间, 下面的代码都省略了静态导入 import static org.mockito.Mockito.*;blog
<dependency> <groupId>org.mockito</groupId> <artifactId>mockito-core</artifactId> <version>2.0.111-beta</version> </dependency>
@Test public void createMockObject() { // 使用 mock 静态方法建立 Mock 对象. List mockedList = mock(List.class); Assert.assertTrue(mockedList instanceof List); // mock 方法不只能够 Mock 接口类, 还能够 Mock 具体的类型. ArrayList mockedArrayList = mock(ArrayList.class); Assert.assertTrue(mockedArrayList instanceof List); Assert.assertTrue(mockedArrayList instanceof ArrayList); }
如上代码所示, 咱们调用了 mock 静态方法来建立一个 Mock 对象. mock 方法接收一个 class 类型, 即咱们须要 mock 的类型.接口
当咱们有了一个 Mock 对象后, 咱们能够定制它的具体的行为. 例如:three
@Test public void configMockObject() { List mockedList = mock(List.class); // 咱们定制了当调用 mockedList.add("one") 时, 返回 true when(mockedList.add("one")).thenReturn(true); // 当调用 mockedList.size() 时, 返回 1 when(mockedList.size()).thenReturn(1); Assert.assertTrue(mockedList.add("one")); // 由于咱们没有定制 add("two"), 所以返回默认值, 即 false. Assert.assertFalse(mockedList.add("two")); Assert.assertEquals(mockedList.size(), 1); Iterator i = mock(Iterator.class); when(i.next()).thenReturn("Hello,").thenReturn("Mockito!"); String result = i.next() + " " + i.next(); //assert Assert.assertEquals("Hello, Mockito!", result); }
咱们使用 when(...).thenReturn(...) 方法链来定义一个行为, 例如 "when(mockedList.add("one")).thenReturn(true)" 表示: 当调用了mockedList.add("one"), 那么返回 true.. 而且要注意的是, when(...).thenReturn(...) 方法链不单单要匹配方法的调用, 并且要方法的参数同样才行.
并且有趣的是, when(...).thenReturn(...) 方法链能够指定多个返回值, 当这样作后, 若是屡次调用指定的方法, 那么这个方法会依次返回这些值. 例如 "when(i.next()).thenReturn("Hello,").thenReturn("Mockito!");", 这句代码表示: 第一次调用 i.next() 时返回 "Hello,", 第二次调用 i.next() 时返回 "Mockito!".
上面的例子咱们展现了方法调用返回值的定制, 那么咱们能够指定一个抛出异常吗? 固然能够的, 例如:
@Test(expected = NoSuchElementException.class) public void testForIOException() throws Exception { Iterator i = mock(Iterator.class); when(i.next()).thenReturn("Hello,").thenReturn("Mockito!"); // 1 String result = i.next() + " " + i.next(); // 2 Assert.assertEquals("Hello, Mockito!", result); doThrow(new NoSuchElementException()).when(i).next(); // 3 i.next(); // 4 }
上面代码的第一第二步咱们已经很熟悉了, 接下来第三部咱们使用了一个新语法: doThrow(ExceptionX).when(x).methodCall
, 它的含义是: 当调用了 x.methodCall 方法后, 抛出异常 ExceptionX.
所以 doThrow(new NoSuchElementException()).when(i).next() 的含义就是: 当第三次调用 i.next() 后, 抛出异常 NoSuchElementException.(由于 i 这个迭代器只有两个元素)
Mockito 会追踪 Mock 对象的所用方法调用和调用方法时所传递的参数. 咱们能够经过 verify() 静态方法来来校验指定的方法调用是否知足断言. 语言描述有一点抽象, 下面咱们仍然以代码来讲明一下.
@Test public void testVerify() { List mockedList = mock(List.class); mockedList.add("one"); mockedList.add("two"); mockedList.add("three times"); mockedList.add("three times"); mockedList.add("three times"); when(mockedList.size()).thenReturn(5); Assert.assertEquals(mockedList.size(), 5); verify(mockedList, atLeastOnce()).add("one"); verify(mockedList, times(1)).add("two"); verify(mockedList, times(3)).add("three times"); verify(mockedList, never()).isEmpty(); }
上面的例子前半部份没有什么特别的, 咱们关注后面的:
verify(mockedList, atLeastOnce()).add("one"); verify(mockedList, times(1)).add("two"); verify(mockedList, times(3)).add("three times"); verify(mockedList, never()).isEmpty();
读者根据代码也应该能够猜想出它的含义了, 很简单:
第一句校验 mockedList.add("one") 至少被调用了 1 次(atLeastOnce)
第二句校验 mockedList.add("two") 被调用了 1 次(times(1))
第三句校验 mockedList.add("three times") 被调用了 3 次(times(3))
第四句校验 mockedList.isEmpty() 从未被调用(never)
Mockito 提供的 spy 方法能够包装一个真实的 Java 对象, 并返回一个包装后的新对象. 若没有特别配置的话, 对这个新对象的全部方法调用, 都会委派给实际的 Java 对象. 例如:
@Test public void testSpy() { List list = new LinkedList(); List spy = spy(list); // 对 spy.size() 进行定制. when(spy.size()).thenReturn(100); spy.add("one"); spy.add("two"); // 由于咱们没有对 get(0), get(1) 方法进行定制, // 所以这些调用实际上是调用的真实对象的方法. Assert.assertEquals(spy.get(0), "one"); Assert.assertEquals(spy.get(1), "two"); Assert.assertEquals(spy.size(), 100); }
这个例子中咱们实例化了一个 LinkedList 对象, 而后使用 spy() 方法对 list 对象进行部分模拟. 接着咱们使用 when(...).thenReturn(...) 方法链来规定 spy.size() 方法返回值是 100. 随后咱们给 spy 添加了两个元素, 而后再 调用 spy.get(0) 获取第一个元素.
这里有意思的地方是: 由于咱们没有定制 add("one"), add("two"), get(0), get(1), 所以经过 spy 调用这些方法时, 其实是委派给 list 对象来调用的.
然而咱们 定义了 spy.size() 的返回值, 所以当调用 spy.size() 时, 返回 100.
Mockito 允准咱们捕获一个 Mock 对象的方法调用所传递的参数, 例如:
@Test public void testCaptureArgument() { List<String> list = Arrays.asList("1", "2"); List mockedList = mock(List.class); ArgumentCaptor<List> argument = ArgumentCaptor.forClass(List.class); mockedList.addAll(list); verify(mockedList).addAll(argument.capture()); Assert.assertEquals(2, argument.getValue().size()); Assert.assertEquals(list, argument.getValue()); }
咱们经过 verify(mockedList).addAll(argument.capture()) 语句来获取 mockedList.addAll 方法所传递的实参 list.
本文由 yongshun 发表于我的博客, 采用署名-非商业性使用-相同方式共享 3.0 中国大陆许可协议.
非商业转载请注明做者及出处. 商业转载请联系做者本人
Email: yongshun1228@gmail.com
本文标题为: 手把手教你 Mockito 的使用
本文连接为: https://segmentfault.com/a/11...