Mockito 应用指南

简介: Mockito 是一个针对 Java 的 mock 框架。

Mockito 是一个针对 Javamock 框架。


预备知识


  如果需要往下学习,你需要先理解 JUnit 框架中的单元测试。


  如果你不熟悉 JUnit,请看 JUnit 教程。


使用 mock 对象来进行测试


单元测试的目标和挑战


  单元测试的思路是在不涉及依赖关系的情况下测试代码(隔离性),所以测试代码与其他类或者系统的关系应该尽量被消除。一个可行的消除方法是替换掉依赖类(测试替换),也就是说我们可以使用替身来替换掉真正的依赖对象。



测试类的分类


  · dummy object 做为参数传递给方法但是绝对不会被使用。譬如说,这种测试类内部的方法不会被调用,或者是用来填充某个方法的参数。


 · Fake 是真正接口或抽象类的实现体,但给对象内部实现很简单。譬如说,它存在内存中而不是真正的数据库中。(Fake 实现了真正的逻辑,但它的存在只是为了测试,而不适合于用在产品中。


 · stub 类是依赖类的部分方法实现,而这些方法在你测试类和接口的时候会被用到,也就是说stub类在测试中会被实例化。stub类会回应任何外部测试的调用。stub类有时候还会记录调用的一些信息。


 · mock object 是指类或者接口的模拟实现,你可以自定义这个对象中某个方法的输出结果。


 测试替代技术能够在测试中模拟测试类以外对象。因此你可以验证测试类是否响应正常。譬如说,你可以验证在Mock对象的某一个方法是否被调用。这可以确保隔离了外部依赖的干扰只测试测试类。


 我们选择Mock对象的原因是因为Mock对象只需要少量代码的配置。


Mock 对象的产生


  你可以手动创建一个 Mock 对象或者使用 Mock 框架来模拟这些类,Mock 框架允许你在运行时创建 Mock 对象并且定义它的行为。


 一个典型的例子是把 Mock 对象模拟成数据的提供者。在正式的生产环境中它会被实现用来连接数据源。但是我们在测试的时候 Mock 对象将会模拟成数据提供者来确保我们的测试环境始终是相同的。


 Mock 对象可以被提供来进行测试。因此,我们测试的类应该避免任何外部数据的强依赖。


 通过 Mock 对象或者 Mock 框架,我们可以测试代码中期望的行为。譬如说,验证只有某个存在 Mock 对象的方法是否被调用了。


使用 Mockito 生成 Mock 对象


  Mockito 是一个流行 mock 框架,可以和 JUnit 结合起来使用。Mockito 允许你创建和配置 mock 对象。使用 Mockito 可以明显的简化对外部依赖的测试类的开发。


  一般使用 Mockito 需要执行下面三步:


  1. 模拟并替换测试代码中外部依赖


  2. 执行测试代码


  3. 验证测试代码是否被正确的执行。


为自己的项目添加 Mockito 依赖


在 Gradle 添加 Mockito 依赖


  如果你的项目使用 Gradle 构建,将下面代码加入 Gradle 的构建文件中为自己项目添加 Mockito 依赖


repositories { jcenter() }
dependencies { testCompile "org.mockito:mockito-core:2.0.57-beta" }



在 Maven 添加 Mockito 依赖


  需要在 Maven 声明依赖,您可以在 http://search.maven.org 网站中搜索 g:“org.mockito”, a:“mockito-core” 来得到具体的声明方式。


在 Eclipse IDE 使用 Mockito


  Eclipse IDE 支持 Gradle 和 Maven 两种构建工具,所以在 Eclipse IDE 添加依赖取决你使用的是哪一个构建工具。


以 OSGi 或者 Eclipse 插件形式添加 Mockito 依赖


 在 Eclipse RCP 应用依赖通常可以在 p2 update 上得到。Orbit 是一个很好的第三方仓库,我们可以在里面寻找能在 Eclipse 上使用的应用和插件。Orbit 仓库地址:http://download.eclipse.org/tools/orbit/downloads


使用 Mockito API


静态引用


  如果在代码中静态引用了org.mockito.Mockito.*;,那你你就可以直接调用静态方法和静态变量而不用创建对象,譬如直接调用 mock() 方法。


使用 Mockito 创建和配置 mock 对象


  除了上面所说的使用 mock() 静态方法外,Mockito 还支持通过 @Mock 注解的方式来创建 mock 对象。


 如果你使用注解,那么必须要实例化 mock 对象。Mockito 在遇到使用注解的字段的时候,会调用MockitoAnnotations.initMocks(this) 来初始化该 mock 对象。另外也可以通过使用@RunWith(MockitoJUnitRunner.class)来达到相同的效果。通过下面的例子我们可以了解到使用@Mock 的方法和MockitoRule规则。


import static org.mockito.Mockito.*;​
public class MockitoTest  {
​        @Mock
        MyDatabase databaseMock; (1)​    
        @Rule public MockitoRule mockitoRule = MockitoJUnit.rule(); (2)
        ​@Test        
        public void testQuery()  {                
          ClassToTest t  = new ClassToTest(databaseMock); (3)                
          boolean check = t.query("* from t"); (4)                
          assertTrue(check); (5)                
          verify(databaseMock).query("* from t"); (6)        
        }
}

 1.告诉 Mockito 模拟 databaseMock 实例


 2.Mockito 通过 @mock 注解创建 mock 对象


 3.使用已经创建的 mock 初始化这个类


 4.在测试环境下,执行测试类中的代码


 5.使用断言确保调用的方法返回值为 true


 6.验证 query 方法是否被 MyDatabase 的 mock 对象调用


配置 mock


  当我们需要配置某个方法的返回值的时候,Mockito 提供了链式的 API 供我们方便的调用


 when(….).thenReturn(….)可以被用来定义当条件满足时函数的返回值,如果你需要定义多个返回值,可以多次定义。当你多次调用函数的时候,Mockito 会根据你定义的先后顺序来返回返回值。Mocks 还可以根据传入参数的不同来定义不同的返回值。譬如说你的函数可以将anyString 或者 anyInt作为输入参数,然后定义其特定的放回值。


import static org.mockito.Mockito.*;
import static org.junit.Assert.*;​
@Test
public void test1()  {
        //  创建 mock        
        MyClass test = Mockito.mock(MyClass.class);​        
        // 自定义 getUniqueId() 的返回值        
        when(test.getUniqueId()).thenReturn(43);​        
        // 在测试中使用mock对象        
        assertEquals(test.getUniqueId(), 43);
}​
// 返回多个值
@Test
public void testMoreThanOneReturnValue()  {
        Iterator i= mock(Iterator.class);        
        when(i.next()).thenReturn("Mockito").thenReturn("rocks");        
        String result=i.next()+" "+i.next();        
        // 断言        
        assertEquals("Mockito rocks", result);
}​
// 如何根据输入来返回值
@Test
public void testReturnValueDependentOnMethodParameter()  {
        Comparable c= mock(Comparable.class);        
        when(c.compareTo("Mockito")).thenReturn(1);        
        when(c.compareTo("Eclipse")).thenReturn(2);        
        // 断言        
        assertEquals(1,c.compareTo("Mockito"));
}​
// 如何让返回值不依赖于输入
@Test
public void testReturnValueInDependentOnMethodParameter()  {
        Comparable c= mock(Comparable.class);        
        when(c.compareTo(anyInt())).thenReturn(-1);        
        // 断言        
        assertEquals(-1 ,c.compareTo(9));
}​
// 根据参数类型来返回值
@Test
public void testReturnValueInDependentOnMethodParameter()  {
        Comparable c= mock(Comparable.class);        
        when(c.compareTo(isA(Todo.class))).thenReturn(0);        
        // 断言        
        Todo todo = new Todo(5);
        assertEquals(todo ,c.compareTo(new Todo(1)));
}


  对于无返回值的函数,我们可以使用doReturn(…).when(…).methodCall来获得类似的效果。例如我们想在调用某些无返回值函数的时候抛出异常,那么可以使用doThrow 方法。如下面代码片段所示


import static org.mockito.Mockito.*;
import static org.junit.Assert.*;​
// 下面测试用例描述了如何使用doThrow()方法​
@Test(expected=IOException.class)
public void testForIOException() {
        // 创建并配置 mock 对象        
        OutputStream mockStream = mock(OutputStream.class);        
        doThrow(new IOException()).when(mockStream).close();​        
        // 使用 mock        
        OutputStreamWriter streamWriter= new OutputStreamWriter(mockStream);        
        streamWriter.close();
}


验证 mock 对象方法是否被调用


  Mockito 会跟踪 mock 对象里面所有的方法和变量。所以我们可以用来验证函数在传入特定参数的时候是否被调用。这种方式的测试称行为测试,行为测试并不会检查函数的返回值,而是检查在传入正确参数时候函数是否被调用。


import static org.mockito.Mockito.*;
​@Test
public void testVerify()  {
        // 创建并配置 mock 对象        
        MyClass test = Mockito.mock(MyClass.class);        
        when(test.getUniqueId()).thenReturn(43);​        
        // 调用mock对象里面的方法并传入参数为12        
        test.testing(12);        
        test.getUniqueId();        
        test.getUniqueId();​        
        // 查看在传入参数为12的时候方法是否被调用        
        verify(test).testing(Matchers.eq(12));​        
        // 方法是否被调用两次        
        verify(test, times(2)).getUniqueId();​        
        // 其他用来验证函数是否被调用的方法        
        verify(mock, never()).someMethod("never called");        
        verify(mock, atLeastOnce()).someMethod("called at least once");        
        verify(mock, atLeast(2)).someMethod("called at least twice");        
        verify(mock, times(5)).someMethod("called five times");        
        verify(mock, atMost(3)).someMethod("called at most 3 times");
}


使用 Spy 封装 java 对象


  @Spy 或者spy()方法可以被用来封装 java 对象。被封装后,除非特殊声明(打桩 stub),否则都会真正的调用对象里面的每一个方法


import static org.mockito.Mockito.*;​
// Lets mock a Linked
ListList list = new LinkedList();
List spy = spy(list);​
// 可用 doReturn() 来打桩
doReturn("foo").when(spy).get(0);​
// 下面代码不生效
// 真正的方法会被调用
// 将会抛出 IndexOutOfBoundsException 的异常,因为 List 为空
when(spy.get(0)).thenReturn("foo");


  方法verifyNoMoreInteractions()允许你检查没有其他的方法被调用了。


使用 @InjectMocks 在 Mockito 中进行依赖注入


  我们也可以使用@InjectMocks 注解来创建对象,它会根据类型来注入对象里面的成员方法和变量。假定我们有 ArticleManager 类


public class ArticleManager {
    private User user;    
    private ArticleDatabase database;​    
    ArticleManager(User user) {
         this.user = user;    
    }​    
    void setDatabase(ArticleDatabase database) { }
}


  这个类会被 Mockito 构造,而类的成员方法和变量都会被 mock 对象所代替,正如下面的代码片段所示:


@RunWith(MockitoJUnitRunner.class)
public class ArticleManagerTest  {
​    @Mock ArticleCalculator calculator;       
  @Mock ArticleDatabase database;       
  @Most User user;​       
  @Spy private UserProvider userProvider = new ConsumerUserProvider();​
  @InjectMocks private ArticleManager manager; (1)​       
  @Test public void shouldDoSomething() {
      // 假定 ArticleManager 有一个叫 initialize() 的方法被调用了               
      // 使用 ArticleListener 来调用 addListener 方法               
      manager.initialize();​               
      // 验证 addListener 方法被调用               
      verify(database).addListener(any(ArticleListener.class));       
  }
}


捕捉参数


  ArgumentCaptor类允许我们在 verification 期间访问方法的参数。得到方法的参数后我们可以使用它进行测试。


import static org.hamcrest.Matchers.hasItem;
import static org.junit.Assert.assertThat;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.verify;​
import java.util.Arrays;
import java.util.List;​
import org.junit.Rule;
import org.junit.Test;
import org.mockito.ArgumentCaptor;
import org.mockito.Captor;
import org.mockito.junit.MockitoJUnit;
import org.mockito.junit.MockitoRule;​
public class MockitoTests {
    @Rule    
    public MockitoRule rule = MockitoJUnit.rule();
    ​@Captor    
    private ArgumentCaptor<List<String>> captor;​    
  @Test    
  public final void shouldContainCertainListItem() {        
    List<String> asList = Arrays.asList("someElement_test", "someElement");        
    final List<String> mockedList = mock(List.class);        
    mockedList.addAll(asList);​        
    verify(mockedList).addAll(captor.capture());        
    final List<String> capturedArgument = captor.getValue();        
    assertThat(capturedArgument, hasItem("someElement"));    
  }
}

Mockito 的限制


  Mockito 当然也有一定的限制。而下面三种数据类型则不能够被测试


  · final classes


  · anonymous classes


  · primitive types


相关文章
|
7月前
|
Java 测试技术 持续交付
Springboot中JUNIT5单元测试+Mockito详解
Springboot中JUNIT5单元测试+Mockito详解
747 1
|
Java 测试技术 Maven
mockito的详细使用
1.概述 mock,一种JAVA单元测试技术,mock允许使用模拟对象替换测试中的系统部件,并断言它们是如何被使用的一项技术。 当某个接口或者功能模块依赖于其他接口或者模块,而所依赖的模块或接口未开发完毕,可以使用mock模拟依赖的模块。 mockito,JAVA单元测试中使用频率最高的mock框架之一。 mock遵循流程:输入—期望—验证
262 0
|
Java 测试技术 Spring
Mockito + Junit + SpringBoot进行单元测试
Mockito + Junit + SpringBoot进行单元测试
263 0
|
监控 Java 测试技术
Mockito教程
Mockito教程
2076 0
Mockito教程
|
Go Java
mockito中两种部分mock的实现,spy、callRealMethod
什么是类的部分mock(partial mock)?A:部分mock是说一个类的方法有些是实际调用,有些是使用mockito的stubbing(桩实现)。   为什么需要部分mock? A:当需要测试一个组合方法(一个方法需要其它多个方法协作)的时候,某个叶子方法(只供别人调用,自己不依赖其它反复)已经被测试过,我们其实不需要再次测试这个叶子方法,so,让叶子打桩实现返回结果,上层方法实际调用并测试。
4242 1
|
Java 测试技术 Maven
PowerMock+Junit的使用
今天使用PowerMock进行单元测试,
160 0
|
敏捷开发 测试技术 数据库
PowerMock(一):PowerMock的基本使用
您好,我是码农飞哥,感谢您阅读本文!本文主要介绍PowerMock的基本使用。
1520 0
|
监控 Java 测试技术
spring单元测试之Mockito
Mockito 是一个针对 Java 的单元测试模拟框架,它与 EasyMock 和 jMock 很相似,都是为了简化单元测试过程中测试上下文 ( 或者称之为测试驱动函数以及桩函数 ) 的搭建而开发的工具
426 0
|
测试技术 容器 API
JUnit + Mockito 单元测试(二)
JUnit 是单元测试框架。Mockito 与 JUnit 不同,并不是单元测试框架(这方面 JUnit 已经足够好了),它是用于生成模拟对象或者直接点说,就是”假对象“的工具。两者定位不同,所以一般通常的做法就是联合 JUnit + Mockito 来进行测试。
2125 0
|
测试技术 API
Mockito 一个优秀的 Mock 测试框架
Hello 大家好,我是阿粉,日常工作中很多时候我们都需要同事间的相互配合协作完成某些功能,所以我们经常会遇到服务或者应用内不同模块之间要互相依赖的场景。比如下面的场景,serviceA 中的 methodA() 方式依赖 serviceB 中的 methodB() 方法返回操作的结果。那如果我们要对自己的methodA() 方法进行编写单元测试,还需要等其他同事的methodB() 方法开发完成才行。那有没有什么办法我们可以跳过或者说模拟方法 B 的输出呢?这就引出了我们今天的主角 Mockito,一个优秀的 Mock 测试框架。
Mockito 一个优秀的 Mock 测试框架