接上一篇,继续学习其它的....

8、找出冗余的互动(即未被验证到的)

@Test(expected = NoInteractionsWanted.class)

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);

List list2 = mock(List.class);

list2.add(1);

list2.add(2);

verify(list2).add(1);

//检查是否有未被验证的互动行为,因为add(2)没有被验证,所以下面的代码会失败抛出异常

verifyNoMoreInteractions(list2);

}

9、使用注解来快速模拟

在上面的测试中我们在每个测试方法里都mock了一个List对象,为了避免重复的mock,使测试类更具有可读性,我们可以使用下面的注解方式来快速模拟对象:

@Mock

private List mockList;

OK,我们再用注解的mock对象试试

@Test

public void shorthand(){

mockList.add(1);

verify(mockList).add(1);

}

运行这个测试类你会发现报错了,mock的对象为NULL,为此我们必须在基类中添加初始化mock的代码

public class MockitoExample2 {

@Mock

private List mockList;

public MockitoExample2(){

MockitoAnnotations.initMocks(this);

}

@Test

public void shorthand(){

mockList.add(1);

verify(mockList).add(1);

}

}

或者使用built-in runner:MockitoJUnitRunner

@RunWith(MockitoJUnitRunner.class)

public class MockitoExample2 {

@Mock

private List mockList;

@Test

public void shorthand(){

mockList.add(1);

verify(mockList).add(1);

}

}

更多的注解还有@Captor,@Spy,@InjectMocks

10、连续调用

@Test(expected = RuntimeException.class)

public void consecutive_calls(){

//模拟连续调用返回期望值,如果分开,则只有最后一个有效

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

public void answer_with_callback(){

//使用Answer来生成我们我们期望的返回

when(mockList.get(anyInt())).thenAnswer(new Answer<Object>() {

@Override

public Object answer(InvocationOnMock invocation) throws Throwable {

Object[] args = invocation.getArguments();

return "hello world:"+args[0];

}

});

assertEquals("hello world:0",mockList.get(0));

assertEquals("hello world:999",mockList.get(999));

}

12、监控真实对象

使用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 unstubbed_invocations(){

//mock对象使用Answer来对未预设的调用返回默认期望值

List mock = mock(List.class,new Answer() {

@Override

public Object answer(InvocationOnMock invocation) throws Throwable {

return 999;

}

});

//下面的get(1)没有预设,通常情况下会返回NULL,但是使用了Answer改变了默认期望值

assertEquals(999, mock.get(1));

//下面的size()没有预设,通常情况下会返回0,但是使用了Answer改变了默认期望值

assertEquals(999,mock.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());

//重置mock,清除所有的互动和预设

reset(list);

assertEquals(0,list.size());

}

Mockito框架入门教程(二)的更多相关文章

  1. Mockito框架入门教程(一)

    官网: http://mockito.org API文档:http://docs.mockito.googlecode.com/hg/org/mockito/Mockito.html 项目源码:htt ...

  2. 无废话ExtJs 入门教程二十一[继承:Extend]

    无废话ExtJs 入门教程二十一[继承:Extend] extjs技术交流,欢迎加群(201926085) 在开发中,我们在使用视图组件时,经常要设置宽度,高度,标题等属性.而这些属性可以通过“继承” ...

  3. 无废话ExtJs 入门教程二十[数据交互:AJAX]

    无废话ExtJs 入门教程二十[数据交互:AJAX] extjs技术交流,欢迎加群(521711109) 1.代码如下: 1 <!DOCTYPE html PUBLIC "-//W3C ...

  4. 无废话ExtJs 入门教程二[Hello World]

    无废话ExtJs 入门教程二[Hello World] extjs技术交流,欢迎加群(201926085) 我们在学校里学习任何一门语言都是从"Hello World"开始,这里我 ...

  5. CodeIgniter框架入门教程——第一课 Hello World!

    本文转载自:http://www.softeng.cn/?p=45 今天开始,我将在这里连载由我自己编写的<CodeIgniter框架入门教程>,首先,这篇教程的读着应该是有PHP基础的编 ...

  6. mongodb入门教程二

    title: mongodb入门教程二 date: 2016-04-07 10:33:02 tags: --- 上一篇文章说了mongodb最基本的东西,这边博文就在深入一点,说一下mongo的一些高 ...

  7. SpringBoot入门教程(二)CentOS部署SpringBoot项目从0到1

    在之前的博文<详解intellij idea搭建SpringBoot>介绍了idea搭建SpringBoot的详细过程, 并在<CentOS安装Tomcat>中介绍了Tomca ...

  8. Java - Struts框架教程 Hibernate框架教程 Spring框架入门教程(新版) sping mvc spring boot spring cloud Mybatis

    https://www.zhihu.com/question/21142149 http://how2j.cn/k/hibernate/hibernate-tutorial/31.html?tid=6 ...

  9. PySide——Python图形化界面入门教程(二)

    PySide——Python图形化界面入门教程(二) ——交互Widget和布局容器 ——Interactive Widgets and Layout Containers 翻译自:http://py ...

随机推荐

  1. TensorFlow基础

    TensorFlow基础 SkySeraph  2017 Email:skyseraph00#163.com 更多精彩请直接访问SkySeraph个人站点:www.skyseraph.com Over ...

  2. Binary Search(Java)(非递归)

    public static int rank(int[] array, int k) { int front = 0, rear = array.length - 1; while(front < ...

  3. Mac之lnmp环境搭建

    之前在Windows上开发大部分都是使用的集成环境(xampp,phpstudy,wamp),可以完成日常便捷开发,有些时候却Windows下无法实现的就需要自己搭建虚拟机,在虚拟机中搭建lnmp环境 ...

  4. Babel插件开发入门指南

    文章概览 主要包括:Babel如何进行转码.插件编写的入门基础.实例讲解如何编写插件. 阅读本文前,需要读者对Babel插件如何使用.配置有一定了解,可以参考笔者之前的文章. 本文所有例子可以在 笔者 ...

  5. Uint 5.css继承权重,盒模型和border padding

    一 .css的继承性和权重 1.1 继承性:继承是CSS的一个主要特征,它是依赖于祖先-后代的关系的.继承是一种机制,它允许样式不仅可以应用于某个特定的元素,还可以应用于它的后代. 可以被继承的属性有 ...

  6. BEX5下实现鼠标滚动滚动条

    使用前提: 页面内容过多,默认的滚动条太难看,在不引入滚动条插件情况下让界面不使用滚动条,又能通过鼠标滚动 实现步骤: 1 在会出现滚动条的组件上设置隐藏滚动条 overflow:hidden; 2 ...

  7. Window下搭建X5本地应用打包服务器

    总的来说就是安装虚拟机,装载VM文件 X5打包服务器(App-Builder)是通过服务方式把X5开发工具(Studio)创建的本地应用进行打包和数字签名,开发者不用单独构建原生代码的编译环境,方便开 ...

  8. 转 spring注解式参数校验

    转自: https://blog.csdn.net/jinzhencs/article/details/51682830 转自: https://blog.csdn.net/zalan01408980 ...

  9. 使用jenkins进行前端项目自动部署

    前面的话 后端的nodeJS项目可以使用pm2进行自动部署,由于前端项目打包后是静态资源,不需要进程守护.一般地,前端项目使用jenkins来进行自动部署,包括打包.测试等一系列流程.本文将详细介绍j ...

  10. python之常用模块二(hashlib logging configparser)

    摘要:hashlib ***** logging ***** configparser * 一.hashlib模块 Python的hashlib提供了常见的摘要算法,如MD5,SHA1等等. 摘要算法 ...