16.3.1  模拟测试概述

目前支持Java语言的Mock测试工具有EasyMock、JMock、Mockito、MockCreator、Mockrunner、MockMaker等,Mockito是一个针对Java的Mocking框架。它与EasyMock和JMock很相似,是一套通过简单的方法对于指定的接口或类生成 Mock 对象的类库,避免了手工编写Mock对象。但Mockito是通过在执行后校验什么已经被调用,它消除了对期望行为(Expectations)的需要。使用Mockito,在准备阶段只需花费很少的时间,可以使用简洁的API编写出漂亮的测试,可以对具体的类创建Mock对象,并且有"监视"非Mock对象的能力。

Mockito使用起来简单,学习成本很低,而且具有非常简洁的API,测试代码的可读性很高,因此它十分受欢迎,用户群越来越多,很多开源软件也选择了Mockito。要想了解更多有关Mockito的信息,可以访问其官方网站http://www.mockito.org/。在开始使用Mockito之前,先简单了解一下Stub和Mock的区别。相比Easymock,JMock,编写出来的代码更加容易阅读。无须录制mock方法调用就返回默认值是一个很大优势。目前最新的版本是1.9.0。

Stub对象用来提供测试时所需要的测试数据,可以对各种交互设置相应的回应。例如我们可以设置方法调用的返回值等。Mockito中 when(…).thenReturn(…) 这样的语法便是设置方法调用的返回值。另外也可以设置方法在何时调用会抛出异常等。

Mock对象用来验证测试中所依赖对象间的交互是否能够达到预期。Mockito中用 verify(…).methodXxx(…) 语法来验证 methodXxx方法是否按照预期进行了调用。有关stub和mock的详细论述请见Martin Fowler的文章《Mocks Aren't Stub》,地址为http://martinfowler.com/articles/mocksArentStubs.html。在Mocking框架中所谓的Mock对象实际上是作为上述的Stub和Mock对象同时使用的。因为它既可以设置方法调用返回值,又可以验证方法的调用。

16.3.2  创建Mock对象

可以对类和接口进行Mock对象的创建,创建的时候可以为Mock对象命名,也可以忽略命名参数。为Mock对象命名的好处就是调试的时候会很方便。比如,我们Mock多个对象,在测试失败的信息中会把有问题的Mock对象打印出来,有了名字我们可以很容易定位和辨认出是哪个Mock对象出现的问题。另外它也有限制,对于final类、匿名类和Java的基本类型是无法进行Mock的。除了用Mock方法来创建模拟对象,如mock(Class<T> classToMock),也可以使用@mock注解定义Mock,下面我们通过实例来介绍一下如何创建一个Mock对象。代码清单16 8  MockitoSampleTest.java创建Mock对象

  1. import org.junit.Test;
  2. import org.mockito.Mock;
  3. import com.baobaotao.domain.User;
  4. import com.baobaotao.service.UserService;
  5. import com.baobaotao.service.UserServiceImpl;
  6. import static org.junit.Assert.*;
  7. import static org.mockito.Mockito.*;
  8. import org.mockito.MockitoAnnotations;
  9. public class MockitoSampleTest{
  10. //① 对接口进行模拟
  11. UserService mockmockUserService = mock(UserService.class);
  12. //② 对类进行模拟
  13. UserServiceImpl mockmockServiceImpl = mock(UserServiceImpl.class);
  14. //③ 基于注解模拟类
  15. @Mock
  16. User mockUser;
  17. @Before
  18. public void initMocks() {
  19. //④ 初始化当前测试类所有@Mock注解模拟对象
  20. MockitoAnnotations.initMocks(this);
  21. }
  22. }

在①处和②处,通过Mockito提供的mock()方法创建UserService 用户服务接口、用户服务实现类UserServiceImpl的模拟对象。在③处,通过@Mock注解创建用户User类模拟对象,并需要在测试类初始化方法中,通过MockitoAnnotations.initMocks()方法初始化当前测试类中所有打上@Mock注解的模拟对象。如果没有执行这一步初始化动作,测试时会报模拟对象为空对象异常。

16.3.3  设定Mock对象的期望行为及返回值

从上文中我们已经知道可以通过when(mock.someMethod()).thenReturn(value)来设定Mock对象的某个方法调用时的返回值,但它也同样有限制条件:对于static和final修饰的方法是无法进行设定的。下面我们通过实例来介绍一下如何调用方法及设定返回值。

代码清单16 9  MockitoSampleTest.java设定模拟对象的行为及返回值

  1. import org.junit.Test;
  2. import org.mockito.Mock;
  3. import com.baobaotao.domain.User;
  4. import com.baobaotao.service.UserService;
  5. import com.baobaotao.service.UserServiceImpl;
  6. public class MockitoSampleTest {
  7. //① 模拟接口UserService测试
  8. @Test
  9. public void testMockInterface() {
  10. //①-1 对方法设定返回值
  11. when(mockUserService.findUserByUserName("tom")).thenReturn(
  12. new User("tom", "1234"));
  13. //①-2 对方法设定返回值
  14. doReturn(true).when(mockServiceImpl).hasMatchUser("tom", "1234");
  15. //①-3 对void方法进行方法预期设定
  16. User u = new User("John", "1234");
  17. doNothing().when(mockUserService).registerUser(u);
  18. //①-4 执行方法调用
  19. User user = mockUserService.findUserByUserName("tom");
  20. boolean isMatch = mockUserService.hasMatchUser("tom","1234");
  21. mockUserService.registerUser(u);
  22. assertNotNull(user);
  23. assertEquals(user.getUserName(), "tom");
  24. assertEquals(isMatch, true);
  25. }
  26. //② 模拟实现类UserServiceImpl测试
  27. @Test
  28. public void testMockClass() {
  29. // 对方法设定返回值
  30. when(mockServiceImpl.findUserByUserName("tom"))
  31. .thenReturn(new User("tom", "1234"));
  32. doReturn(true).when(mockServiceImpl).hasMatchUser("tom", "1234");
  33. User user = mockServiceImpl.findUserByUserName("tom");
  34. boolean isMatch = mockServiceImpl.hasMatchUser("tom","1234");
  35. assertNotNull(user);
  36. assertEquals(user.getUserName(), "tom");
  37. assertEquals(isMatch, true);
  38. }
  39. //③ 模拟User类测试
  40. @Test
  41. public void testMockUser() {
  42. when(mockUser.getUserId()).thenReturn(1);
  43. when(mockUser.getUserName()).thenReturn("tom");
  44. assertEquals(mockUser.getUserId(),1);
  45. assertEquals(mockUser.getUserName(), "tom");
  46. }

…在①处,模拟测试接口UserService的findUserByUserName()方法、hasMatchUser()方法及registerUser()方法。在①-1处通过when().thenReturn()语法,模拟方法调用及设置方法的返回值,实例通过模拟调用UserService 用户服务接口的查找用户findUserByUserName()方法,查询用户名为"tom"详细的信息,并设置返回User对象:new User("tom", "1234")。在①-2处通过doReturn (). when ()语法,模拟判断用户hasMatchUser()方法的调用,判断用户名为"tom"及密码为"1234"的用户存在,并设置返回值为:true。在①-3处对void方法进行方法预期设定,如实例中调用注册用户registerUser()方法。设定调用方法及返回值之后,就可以执行接口方法调用验证。在②处和③处,模拟测试用户服务实现类UserServiceImpl,测试的方法与模拟接口一致。

16.3.4  验证交互行为

Mock对象一旦建立便会自动记录自己的交互行为,所以我们可以有选择地对其交互行为进行验证。在Mockito中验证mock对象交互行为的方法是verify(mock). xxx()。于是用此方法验证了findUserByUserName()方法的调用,因为只调用了一次,所以在verify中我们指定了times参数或atLeastOnce()参数。最后验证返回值是否和预期一样。

代码清单16 10 MockitoSampleTest.java验证交互行为

  1. import org.junit.Test;
  2. import org.mockito.Mock;
  3. import com.baobaotao.domain.User;
  4. import com.baobaotao.service.UserService;
  5. import com.baobaotao.service.UserServiceImpl;
  6. public class MockitoSampleTest {
  7. //① 模拟接口UserService测试
  8. @Test
  9. public void testMockInterface() {
  10. when(mockUserService.findUserByUserName("tom"))
  11. .thenReturn(new User("tom", "1234"));
  12. User user = mockServiceImpl.findUserByUserName("tom");
  13. //①-4 验证返回值
  14. assertNotNull(user);
  15. assertEquals(user.getUserName(), "tom");
  16. assertEquals(isMatch, true);
  17. //①-5 验证交互行为
  18. verify(mockUserService).findUserByUserName("tom");
  19. //①-6 验证方法至少调用一次
  20. verify(mockUserService, atLeastOnce()).findUserByUserName("tom");
  21. verify(mockUserService, atLeast(1)).findUserByUserName("tom");
  22. //①-7 验证方法至多调用一次
  23. verify(mockUserService, atMost(1)).findUserByUserName("tom");
  24. }

Mockio为我们提供了丰富调用方法次数的验证机制,如被调用了特定次数verify(xxx, times(x))、至少x次verify(xxx, atLeast (x))、最多x次verify(xxx, atMost (x))、从未被调用verify(xxx, never())。在①-6处,验证findUserByUserName()方法至少被调用一次。在①-7处,验证findUserByUserName()方法至多被调用一次。

模拟利器Mockito的更多相关文章

  1. 提升单元测试体验的利器--Mockito使用总结

    为神马要使用Mockito? 在编写单元测试的时候,为了尽可能的保证隔离性,我们时常需要对某些不容易构造或者不容易获取或者对外部环境有依赖的对象,用一个虚拟的对象来创建以便于测试.假设你正在开发的的代 ...

  2. 单元测试利器Mockito框架

    什么是Mock Mock 的中文译为仿制的,模拟的,虚假的.对于测试框架来说,即构造出一个模拟/虚假的对象,使我们的测试能顺利进行下去. Mock 测试就是在测试过程中,对于某些 不容易构造(如 Ht ...

  3. 《Spring 3.x 企业应用开发实战》目录

    图书信息:陈雄华 林开雄 编著 ISBN 978-7-121-15213-9 概述: 第1章:对Spring框架进行宏观性的概述,力图使读者建立起对Spring整体性的认识. 第2章:通过一个简单的例 ...

  4. 五、Spring ——单元测试

    1.JUnit4 JUnit测试用例的完整生命周期要经历一下阶段:类级初始化资源处理,方法级初始化资源处理.执行测试用例中的方法.方法级销毁资源处理.类级销毁资源处理. 测试方法 @Test 初始化 ...

  5. 单元测试系列:Mock工具之Mockito实战

    更多原创测试技术文章同步更新到微信公众号 :三国测,敬请扫码关注个人的微信号,感谢! 原文链接:http://www.cnblogs.com/zishi/p/6780719.html 在实际项目中写单 ...

  6. 单元测试系列之五:Mock工具之Mockito实战

    更多原创测试技术文章同步更新到微信公众号 :三国测,敬请扫码关注个人的微信号,感谢! 原文链接:http://www.cnblogs.com/zishi/p/6780719.html 在实际项目中写单 ...

  7. 使用 Mockito 单元测试 – 教程

    tanyuanji@126.com 版本历史 - - - - 使用 Mockito 进行测试 该教程主要讲解 Mockito 框架在Eclipse IDE 中的使用   目录 tanyuanji@12 ...

  8. Mockito简介(转)

    Mockito 是目前 java 单测中使用比较流行的 mock 工具.其他还有 EasyMock,JMock,MockCreator,Mockrunner,MockMaker 及 PowerMock ...

  9. 手把手教你 Mockito 的使用

    什么是 Mockito Mockito 是一个强大的用于 Java 开发的模拟测试框架, 通过 Mockito 我们可以创建和配置 Mock 对象, 进而简化有外部依赖的类的测试.使用 Mockito ...

随机推荐

  1. JavaWeb笔记(二)Servlet

    Tomcat目录简介 bin--可执行文件 conf--配置文件 lib--依赖jar包 logs--日志文件 temp--临时文件 webapps--默认项目部署路径 work--存放运行时的数据 ...

  2. 使用CORS解决flask前端页面跨域问题

    from flask import Flask from flask_cors import CORS app = Flask(__name__) CORS(app) @app.route(" ...

  3. Codeforces Round #418 (Div. 2) A+B+C!

    终判才知道自己失了智.本场据说是chinese专场,可是请允许我吐槽一下题意! A. An abandoned sentiment from past shabi贪心手残for循环边界写错了竟然还过了 ...

  4. 【bzoj2406】矩阵 二分+有上下界可行流

    题目描述 输入 第一行两个数n.m,表示矩阵的大小. 接下来n行,每行m列,描述矩阵A. 最后一行两个数L,R. 输出 第一行,输出最小的答案: 样例输入 2 2 0 1 2 1 0 1 样例输出 1 ...

  5. libcmt.lib和msvcrt.lib冲突,原因和解决方法

    libcmt.lib和msvcrt.lib冲突,原因和解决方法 https://blog.csdn.net/longlijun/article/details/7331093 libcmt.lib是w ...

  6. HDU 5884 Sort(二分答案+计算WPL的技巧)

    Sort Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submi ...

  7. Java众神之路(2)-标志符

    标志符 1.1标志符:用来标志类名.变量名.方法名.类型名.文件名的有效字符序列成为标志符. 1.2命名规则: Java语言规定标志符由字母(a-zA-Z).下划线(_).美元符号($)和数字(0-9 ...

  8. git下载其他人分支的代码

    1. 在工作空间,右键,打开Git Bash 2. clone主分支的代码(即下载主分支代码的过程) 执行命令: git clone xxx.git 3. 进入工程目录 cd   xxx 4. 切换到 ...

  9. LOJ#2307. 「NOI2017」分身术

    $n \leq 100000$个点,$m \leq 100000$次询问,每次问删掉一些点后的凸包面积. 不会啦写个20暴力,其实是可以写到50的.当个计算几何板子练习. //#include< ...

  10. [LeetCode] Minimum Depth of Binary Tree 二叉树最小深度

    Given a binary tree, find its minimum depth. The minimum depth is the number of nodes along the shor ...