Mockito的使用一

一、Mockito的引用
Gradle:

repositories { jcenter() }
dependencies { testCompile "org.mockito:mockito-core:1.+" }

Maven:

<dependency>
    <groupId>org.mockitogroupId>
    <artifactId>mockito-coreartifactId>
    <version>1.10.19version>
dependency>

二、使用
详情请参考:http://mockito.org/

1、验证某些行为:

//静态导入会使代码更简洁
 import static org.mockito.Mockito.*;

 //创建Mock对象
 List mockedList = mock(List.class);

 //使用 Mock对象
 mockedList.add("one");
 mockedList.clear();

 //验证
 verify(mockedList).add("one");
 verify(mockedList).clear();

其中 verify 方法 还有2个参数的方法:第二个参数可以传入 times(x),atLeastOne() 、never()、atMost(x)等,这些参数表示验证方法执行的次数。
如:

    {
     // 验证方法被调用次数
        LinkedList mockedList = mock(LinkedList.class);
        mockedList.add("one");

        mockedList.add("two");
        mockedList.add("two");

        mockedList.add("three");
        mockedList.add("three");
        mockedList.add("three");
    //验证 add("one")执行了1次
        verify(mockedList,times(2)).add("two");
        //验证 add("two")执行了2次
        verify(mockedList,times(3)).add("three");
    //验证 add("never")执行了3次
        verify(mockedList,never()).add("never");
        //验证 add("one")最少执行了1次
        verify(mockedList,atLeast(1)).add("one");
        //验证 add("three")最多执行了10次
        verify(mockedList,atMost(10)).add("three");

    }

如果第二个参数不写,相当于执行了 verify(mockedList,times(1))。

2、如何测试一些桩

//你可以mock具体的类型,不仅只是接口
 LinkedList mockedList = mock(LinkedList.class);

 //测试桩,在调用mockedList.get(0)时会返回 "first"
 when(mockedList.get(0)).thenReturn("first");
 //测试桩,在调用mockedList.get(1)时 会抛出RuntimeException异常
 when(mockedList.get(1)).thenThrow(new RuntimeException());

 //会打印出 "first"
 System.out.println(mockedList.get(0));

 //会抛出runtime exception
 System.out.println(mockedList.get(1));

 //mockedList.get(999)没有打桩 ,所以会打印出  "null" 
 System.out.println(mockedList.get(999));

3、参数匹配器

//使用内置的anyInt()参数匹配器,当调用get(整数值)时,都返回"element"
 when(mockedList.get(anyInt())).thenReturn("element");

 //使用自定义的参数匹配器,(在 isValid()函数中返回你自己的匹配器实现)
 when(mockedList.contains(argThat(isValid()))).thenReturn("element");

 //打印 "element"
 System.out.println(mockedList.get(999));

// 你也可以验证参数匹配器
 verify(mockedList).get(anyInt());

有一点需要注意,如果使用参数匹配器,那么所有的参数都因该有匹配器提供。
例如:

 //这个是正确的,因为 eq()也是一个参数匹配器
 verify(mock).someMethod(anyInt(), anyString(), eq("third argument"));

  //下面的代码是错误的,因为所有参数必须由匹配器提供,二参数"third argument"并非由参数//匹配器提供
   verify(mock).someMethod(anyInt(), anyString(), "third argument");

4、验证函数的执行次数

//using mock
 mockedList.add("once");

 mockedList.add("twice");
 mockedList.add("twice");

 mockedList.add("three times");
 mockedList.add("three times");
 mockedList.add("three times");

 //following two verifications work exactly the same - times(1) is used by default
 verify(mockedList).add("once");
 verify(mockedList, times(1)).add("once");

 //exact number of invocations verification
 verify(mockedList, times(2)).add("twice");
 verify(mockedList, times(3)).add("three times");

 //verification using never(). never() is an alias to times(0)
 verify(mockedList, never()).add("never happened");

 //verification using atLeast()/atMost()
 verify(mockedList, atLeastOnce()).add("three times");
 verify(mockedList, atLeast(2)).add("five times");
 verify(mockedList, atMost(5)).add("three times");

这些方法在将第一条的时候,已经提过了,这里就不在说明了。

5、为无返回值对象打桩抛出异常


    //当调用mockedList.clear()时,将抛出RuntimeException 异常
   doThrow(new RuntimeException()).when(mockedList).clear();

   //将抛出RuntimeException:
   mockedList.clear();

6、验证方法调用顺序:

 // A.当只有一个mock对象时
        List singleMock = mock(List.class);

        //使用单个mock对象
        singleMock.add("was added first");
        singleMock.add("was added second");

        //创建一个InOrder
        InOrder inOrder = Mockito.inOrder(singleMock);

      //确保add("was added first") 在add("was added second")之前
        inOrder.verify(singleMock).add("was added first");
        inOrder.verify(singleMock).add("was added second");

        // B. 多个Mock对象时
        List firstMock = mock(List.class);
        List secondMock = mock(List.class);

        //使用mock对象那个
        firstMock.add("was called first");
        secondMock.add("was called second");

        //create inOrder object passing any mocks that need to be verified in order
        InOrder inOrder = Mockito.inOrder(firstMock, secondMock);

        //确保firstMock.add("was called firs") 在 secondMock.add("was called second")之前
        inOrder.verify(firstMock).add("was called first");
        inOrder.verify(secondMock).add("was called second");

7、确保交互操作没有执行在Mock对象上

     List mockOne = mock(List.class);
        //使用Mock对新娘
        mockOne.add("one");

        //普通验证
        verify(mockOne).add("one");

        //验证 add("two") 没有执行过
        verify(mockOne, never()).add("two");

        List mockTwo = mock(List.class);
        List mockThree = mock(List.class);

        //验证mockTwo和mockThree 没有交互过
        verifyZeroInteractions(mockTwo, mockThree);

8、寻找多余的调用:

    List mockedList = mock(List.class);
        //常见mock对象
        mockedList.add("one");

        mockedList.add("two");
    //这里只验证了 add("one")没有验证add("two"),这将导致 //verifyNoMoreInteractions(mockedList);验证失败
        verify(mockedList).add("one");

        //这里将验证失败
        verifyNoMoreInteractions(mockedList);

9,简化Mock对象的创建
使用@Mock注解,它的好处在于:
1、最小化重复地创建代码
2、是测试类的代码可读性更高
3、是验证错误更易于阅读,因为字段名可以表示Mock对象

 public class ArticleManagerTest {

       @Mock private ArticleCalculator calculator;
       @Mock private ArticleDatabase database;
       @Mock private UserProvider userProvider;

       private ArticleManager manager;

注意,必须在调用 MockitoAnnotations.initMocks(testClass); 进行初始化。
例如可以在基类,或者当前类

 public class ArticleManagerTest extends SampleBaseTestCase { 

       @Mock private ArticleCalculator calculator;
       @Mock private ArticleDatabase database;
       @Mock private UserProvider userProvider;

       private ArticleManager manager;

       @Before public void setup() {
           manager = new ArticleManager(userProvider, database, calculator);
       }
   }

   public class SampleBaseTestCase {

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

10、为连续调用做测试桩

when(mock.someMethod("some arg"))
   .thenThrow(new RuntimeException())
   .thenReturn("foo");

 //第一次调用时: 抛出RuntimeException异常
 mock.someMethod("some arg");

 //第二次调用时,打印 "foo"
 System.out.println(mock.someMethod("some arg"));

 //后续的调用也是 输出"foo"
 System.out.println(mock.someMethod("some arg"));

还有一种简短版本:


 when(mock.someMethod("some arg"))
   .thenReturn("one", "two", "three");

你可能感兴趣的:(Android,测试,Andorid应用开发)