mockito框架
2016-04-09 15:56:26
参考自
http://www.cnblogs.com/silence-hust/p/5017233.html
http://blog.csdn.net/sdyy321/article/details/38757135
首先创建一个maven工程

在pom文件中,存在如下依赖
<dependencies>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.11</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.mockito</groupId>
<artifactId>mockito-all</artifactId>
<version>1.9.5</version>
<scope>test</scope>
</dependency>
</dependencies>
1、验证行为是否发生
@Test
public void mockedList(){
List mockedList = mock(List.class);
mockedList.add("one");
mockedList.clear();
//验证add("one")和clear()行为是否发生
verify(mockedList).add("one");
verify(mockedList).clear();
}
2、验证返回值
@Test
public void two(){
//模拟创建一个List对象
LinkedList mockLinkedList = mock(LinkedList.class); //打桩,当LinkedList调用get(0)方法时,第一次返回hello,第二次n次返回world
when(mockLinkedList.get(0)).thenReturn("hello").thenReturn("world");
String result = mockLinkedList.get(0) + " " + mockLinkedList.get(0) + " " + mockLinkedList.get(0); //使用mock对象
System.out.println(mockLinkedList.get(0));
System.out.println(mockLinkedList.get(0));
System.out.println(mockLinkedList.get(0)); assertEquals("hello world world",result);
}
这里注意所有的方法都会有返回值,如果没有设置返回值,那么就会返回null或者空集、适当的类型。 Stubbing可以被重写,也就是同一个参数方法可以放回不同的值,但是以最后一次设置的值为标准。一旦被 Stubbed,无论方法被调用多少次,都只会返回Stubbed value。最后一次最重要原则。
第一次运行打印 hello world world,这是第二次运行测试打印 world world world。
3、参数匹配
@Test
public void with_arguments(){
Comparable comparable = mock(Comparable.class);
//预设根据不同的参数返回不同的结果
when(comparable.compareTo("Test")).thenReturn(1);
when(comparable.compareTo("Omg")).thenReturn(2);
assertEquals(1, comparable.compareTo("Test"));
assertEquals(2, comparable.compareTo("Omg"));
//对于没有预设的情况会返回默认值
assertEquals(0, comparable.compareTo("Not stub"));
}
除了匹配给定参数外,还可以使用参数匹配器 argumentMatchers,更加灵活
@Test
public void argumentMatchersTest(){
List list = mock(List.class);
//匹配任意int类型的值
when(list.get(anyInt())).thenReturn(1);
assertEquals(1, list.get(1));
verify(list).get(anyInt());
assertEquals(1, list.get(999));
}
若方法中的某一个参数使用了matcher,则所有的参数都必须使用matcher:
@Test
public void argumentMatchersTest2(){
Map map = mock(Map.class);
//匹配任意参数
when(map.put(anyInt(),anyString())).thenReturn("world");
assertEquals("world", map.put(1,"hello"));
//以下三种验证都可通过测试
verify(map).put(anyInt(),anyString());
verify(map).put(anyInt(),eq("hello"));
verify(map).put(eq(1),eq("hello"));
}
4、验证调用次数
@Test
public void three(){
List mockedList = mock(List.class);
mockedList.add(1);
mockedList.add(2);
mockedList.add(2);
mockedList.add(3);
mockedList.add(3);
mockedList.add(3);
//验证是否被调用一次,等效于下面的times(1),默认的,可以不写
verify(mockedList).add(1);
verify(mockedList,times(1)).add(1);
//验证是否被调用2次
verify(mockedList,times(2)).add(2);
//验证是否被调用3次
verify(mockedList,times(3)).add(3);
//验证是否从未被调用过
verify(mockedList,never()).add(4);
//验证至少调用一次
verify(mockedList,atLeastOnce()).add(1);
//验证至少调用2次
verify(mockedList,atLeast(2)).add(2);
//验证至多调用3次
verify(mockedList,atMost(3)).add(3);
}
5、模拟抛出异常
@Test(expected = IOException.class)
public void when_thenThrow() throws IOException {
OutputStream outputStream = mock(OutputStream.class); //预设当流关闭时抛出异常 ,因为close()返回类型为void,采用了doThrow 形式
doThrow(new IOException()).when(outputStream).close();
outputStream.close();
}
对于返回类型为void的方法,有系列函数可以用来处理,如doThrow, doAnswer, doNothing, doReturn。
6、验证执行的顺序
@Test
public void four(){
List firstList = mock(List.class);
List secondList = mock(List.class);
//using mocks
firstList.add("was called first one mock");
firstList.add("was called second one mock");
secondList.add("was called third one mock");
secondList.add("was called fourth one mock");
//create inOrder object passing any mocks that need to be verified in order // InOrder indOrder = inOrder(firstList,secondList);
InOrder indOrder = inOrder(secondList,firstList);
//实际上两种顺序都通过测试了 indOrder.verify(firstList).add("was called first one mock");
indOrder.verify(firstList).add("was called second one mock");
indOrder.verify(secondList).add("was called third one mock");
indOrder.verify(secondList).add("was called fourth one mock"); }
7、验证零互动
@Test
public void verify_interaction(){
List list = mock(List.class);
List list2 = mock(List.class);
List list3 = mock(List.class);
list.add(1);
verify(list).add(1);
verify(list,never()).add(2);
//验证零互动行为
verifyZeroInteractions(list2,list3);
}
8、找出冗余的互动(即未被验证到的)
@Test
public void find_redundant_interaction(){
List list = mock(List.class);
list.add(1);
list.add(2);
verify(list,times(2)).add(anyInt());
//检查是否有未被验证的互动行为,因为add(1)和add(2)都会被上面的anyInt()验证到,所以下面的代码会通过
verifyNoMoreInteractions(list); //下面因为只验证了add(1),add(2)没有被验证,所以下面的代码会失败抛出异常
// List list2 = mock(List.class);
// list2.add(1);
// list2.add(2);
//verify(list2).add(1);
// verifyNoMoreInteractions(list2);
}
9、使用注解来快速模拟
在上面的测试中我们在每个测试方法里都mock了一个List对象,为了避免重复的mock,是测试类更具有可读性,我们可以使用下面的注解方式来快速模拟对象:
public class MockDemoTest2 {
@Mock
private List mockList;
public MockDemoTest2(){
MockitoAnnotations.initMocks(this);
}
@Test
public void shorthand(){
mockList.add(1);
verify(mockList).add(1);
}
}
注意要在构造函数中初试化mock对象,否则mock对象为null。也可以通过在类上使用注解:@RunWith(MockitoJUnitRunner.class) ,这样就不需要初始化mock了。
@RunWith(MockitoJUnitRunner.class)
public class MockDemoTest2 { @Mock
private List mockList; /* public MockDemoTest2(){
MockitoAnnotations.initMocks(this);
}
*/
@Test
public void shorthand(){
mockList.add(1);
verify(mockList).add(1);
} }
10、连续调用
@Test(expected = RuntimeException.class)
public void consecutive_calls(){
List mockList = mock(List.class);
//模拟连续调用返回期望值,如果分开,则只有最后一个有效
when(mockList.get(0)).thenReturn(0);
when(mockList.get(0)).thenReturn(1);
when(mockList.get(0)).thenReturn(2);
when(mockList.get(1)).thenReturn(0).thenReturn(1).thenThrow(new RuntimeException());
assertEquals(2,mockList.get(0));
assertEquals(2,mockList.get(0));
assertEquals(0,mockList.get(1));
assertEquals(1,mockList.get(1));
//第三次或更多调用都会抛出异常
mockList.get(1);
}
11、使用回调生成期望值
@Test(expected = RuntimeException.class)
public void consecutive_calls(){
List mockList = mock(List.class);
//模拟连续调用返回期望值,如果分开,则只有最后一个有效
when(mockList.get(0)).thenReturn(0);
when(mockList.get(0)).thenReturn(1);
when(mockList.get(0)).thenReturn(2);
when(mockList.get(1)).thenReturn(0).thenReturn(1).thenThrow(new RuntimeException());
assertEquals(2,mockList.get(0));
assertEquals(2,mockList.get(0));
assertEquals(0,mockList.get(1));
assertEquals(1,mockList.get(1));
//第三次或更多调用都会抛出异常
mockList.get(1);
}
12、监控真实对象
当使用spy的时候真正的方法将会被调用,而不再是stub的对象了,这个和部分mock的思想是一样的。
@Test
public void seven(){
List list = new LinkedList();
List spy = spy(list);
//optionally, you can stub out some methods:
when(spy.size()).thenReturn(100);
//using the spy calls real methods
spy.add("one");
spy.add("two");
//prints "one" - the first element of a list
System.out.println(spy.get(0));
//size() method was stubbed - 100 is printed
System.out.println(spy.size());
//optionally, you can verify
verify(spy).add("one");
verify(spy).add("two");
}
使用spy的时候需要注意一点:有时不能用when-then语句,而改用do-when语句
@Test(expected = IndexOutOfBoundsException.class)
public void spy_on_real_objects(){
List list = new LinkedList();
List spy = spy(list);
//下面预设的spy.get(0)会报错,因为会调用真实对象的get(0),所以会抛出越界异常
//when(spy.get(0)).thenReturn(3); //使用doReturn-when可以避免when-thenReturn调用真实对象api
doReturn(999).when(spy).get(999);
//预设size()期望值
when(spy.size()).thenReturn(100);
//调用真实对象的api
spy.add(1);
spy.add(2);
assertEquals(100,spy.size());
assertEquals(1,spy.get(0));
assertEquals(2,spy.get(1));
verify(spy).add(1);
verify(spy).add(2);
assertEquals(999,spy.get(999));
spy.get(2);
}
13、修改对未预设的调用返回默认期望值
@Test
public void eight(){
//mock对象使用Answer来对未预设的调用返回默认期望值
List mocklist = mock(List.class,new Answer(){
public Object answer(InvocationOnMock invocation) throws Throwable {
return 999;
}
});
//下面的get(1)没有预设,通常情况下会返回NULL,但是使用了Answer改变了默认期望值
assertEquals(999, mocklist.get(1));
//下面的size()没有预设,通常情况下会返回0,但是使用了Answer改变了默认期望值
assertEquals(999,mocklist.size());
}
14、捕获参数来进一步断言
@Test
public void capturing_args(){
PersonDao personDao = mock(PersonDao.class);
PersonService personService = new PersonService(personDao); ArgumentCaptor<Person> argument = ArgumentCaptor.forClass(Person.class);
personService.update(1,"jack");
verify(personDao).update(argument.capture());
assertEquals(1,argument.getValue().getId());
assertEquals("jack",argument.getValue().getName());
} class Person{
private int id;
private String name; Person(int id, String name) {
this.id = id;
this.name = name;
} public int getId() {
return id;
} public String getName() {
return name;
}
} interface PersonDao{
public void update(Person person);
} class PersonService{
private PersonDao personDao; PersonService(PersonDao personDao) {
this.personDao = personDao;
} public void update(int id,String name){
personDao.update(new Person(id,name));
}
}
15、真实的部分mock
@Test
public void real_partial_mock(){
//通过spy来调用真实的api
List list = spy(new ArrayList());
assertEquals(0,list.size());
A a = mock(A.class);
//通过thenCallRealMethod来调用真实的api
when(a.doSomething(anyInt())).thenCallRealMethod();
assertEquals(999,a.doSomething(999));
} class A{
public int doSomething(int i){
return i;
}
}
16、重置mock
@Test
public void reset_mock(){
List list = mock(List.class);
when(list.size()).thenReturn(10);
list.add(1);
assertEquals(10,list.size());
verify(list).add(1);
//重置mock,清除所有的互动和预设
reset(list);
assertEquals(0,list.size());
}
mockito框架的更多相关文章
- Mockito框架入门教程(一)
官网: http://mockito.org API文档:http://docs.mockito.googlecode.com/hg/org/mockito/Mockito.html 项目源码:htt ...
- 单元测试利器Mockito框架
什么是Mock Mock 的中文译为仿制的,模拟的,虚假的.对于测试框架来说,即构造出一个模拟/虚假的对象,使我们的测试能顺利进行下去. Mock 测试就是在测试过程中,对于某些 不容易构造(如 Ht ...
- Mockito框架入门教程(二)
接上一篇,继续学习其它的.... 8.找出冗余的互动(即未被验证到的) @Test(expected = NoInteractionsWanted.class) public void find_re ...
- Mockito:一个强大的用于Java开发的模拟测试框架
https://blog.csdn.net/zhoudaxia/article/details/33056093 介绍 本文将介绍模拟测试框架Mockito的一些基础概念, 介绍该框架的优点,讲解应用 ...
- 使用强大的 Mockito 测试框架来测试你的代码
原文链接 : Unit tests with Mockito - Tutorial 译文出自 : 掘金翻译计划 译者 : edvardhua 校对者: hackerkevin, futureshine ...
- Java测试框架Mockito源码分析
1.Mockito简介 测试驱动的开发(Test Driven Design, TDD)要求我们先写单元测试,再写实现代码.在写单元测试的过程中,一个很普遍的问题是,要测试的类会有很多依赖,这些依赖的 ...
- 一文让你快速上手 Mockito 单元测试框架
前言 在计算机编程中,单元测试是一种软件测试方法,通过该方法可以测试源代码的各个单元功能是否适合使用.为代码编写单元测试有很多好处,包括可以及早的发现代码错误,促进更改,简化集成,方便代码重构以及许多 ...
- Mockito测试
Mockito 一 mockito基本概念 Mock测试是单元测试的重要方法之一,而Mockito作为一个流行的Mock框架,简单易学,且有非常简洁的API,测试代码的可读性很高. Mock测试就是在 ...
- 使用 Mockito 单元测试 – 教程
tanyuanji@126.com 版本历史 - - - - 使用 Mockito 进行测试 该教程主要讲解 Mockito 框架在Eclipse IDE 中的使用 目录 tanyuanji@12 ...
随机推荐
- Android studio jni
首先我们要明确几个概念,jni,ndk,共享库(.so). jni是java native interface的缩写,java 本地接口.它提供了若干的API实现了Java和其他语言的通信(主要是C/ ...
- 一、重写(覆盖)override
一.重写(覆盖)override 子类可以继承父类对象的方法,在继承后,重复提供该方法,就叫做方法的重写,又叫做覆盖override package property; //父类对象 public c ...
- Python的数据类型1数值和字符串
Python的交互器 在讲这个之前,我要先讲一下python的交互器,之后讲解代码(除了做简单项目)都在交互器上进行,这样可能比较直接一些,有利于刚接触python的人理解代码 python在命令行自 ...
- oracle 产生一个任意大小的随机数
SELECT DBMS_RANDOM.RANDOM FROM DUAL; 产生一个任意大小的随机数 SELECT ABS(MOD(DBMS_RANDOM.RANDOM,100)) F ...
- winRAR显示树树目录
这样 比较方便
- Java——IO类,转换流简化写法
body, table{font-family: 微软雅黑} table{border-collapse: collapse; border: solid gray; border-width: 2p ...
- CUDA ---- Kernel性能调节
Exposing Parallelism 这部分主要介绍并行分析,涉及掌握nvprof的几个metric参数,具体的这些调节为什么会影响性能会在后续博文解释. 代码准备 下面是我们的kernel函数s ...
- android机顶盒真机调试方法
最近接触电视APP开发,之前对Android开发也不太了解还一直以为不能真机调试.最近静下心来想一想肯定能真机调试的,我是我不知道而已.现在讲述一下真机调试的步骤: 1.进入设置--关于,连续点击版本 ...
- 2019.1.22 zigbee test
1传输测试 频谱仪设置: sigfox 模块串口设置: 自动选择对应型号 Test step: PS:发送TX指令 AT$cw=波特率,通道,uint 这里有个问题--不应该只发送一次 ------- ...
- ztree树形菜单的增加删除修改和换图标
首先需要注意一点,如果有研究过树形菜单,就会发现实现删除和修改功能特别简单,但是增加却有一点复杂.造成这个现象是ztree树形菜单的历史遗留问题.大概是之前的版本没有增加这个功能,后来的版本加上了这个 ...