`

Mockito:一个强大的用于Java开发的模拟测试框架

阅读更多

介绍

 

  本文将介绍模拟测试框架Mockito的一些基础概念, 介绍该框架的优点,讲解应用Mockito的Java示例。

 

模拟(Mock)的概念 

 

  在软件开发的世界之外, "mock"一词是指模仿或者效仿。因此可以将“mock”理解为一个替身,替代者。在软件开发中提及"mock",通常理解为模拟对象或者fake。

  译者注:mock等多代表的是对被模拟对象的抽象类,你可以把fake理解为mock的实例。不知道这样说准不准确:)

 

  Fake通常被用作被测类依赖关系的替代者.。

名词定义  
依赖关系 – 依赖关系是指在应用程序中一个类基于另一个类来执行其预定的功能。依赖关系通常都存在于所依赖的类的实例变量中。
 
被测类 – 在编写单元测试的时候,“单元”一词通常代表一个单独的类及为其编写的测试代码。被测类指的就是其中被测试的类。

 

为什么需要模拟? 

先来看看下面这个示例:

从上图可以看出如果我们要对A进行测试,那么就要先把整个依赖树构建出来,也就是BCDE的实例。

一种替代方案就是使用mock:

 

从图中可以清晰的看出,mock对象就是在调试期间用来作为真实对象的替代品。mock测试就是在测试过程中,对那些不容易构建的对象用一个虚拟对象来代替测试的方法就叫mock测试。

 

  在我们一开始学编程时,我们所写的对象通常都是独立的。hello world之类的类并不依赖其他的类(System.out除外),也不会操作别的类。但实际上软件中是充满依赖关系的。我们会基于service类写操作类,而service类又是基于数据访问类(DAOs)的,依次下去。

图1 类的依赖关系

 

  单元测试的思路就是我们想在不涉及依赖关系的情况下测试代码。这种测试可以让你无视代码的依赖关系去测试代码的有效性。核心思想就是如果代码按设计正常工作,并且依赖关系也正常,那么他们应该会同时工作正常。

  下面的代码就是这样的例子:

[java] view plain copy
 
  1. import java.util.ArrayList;  
  2. public class Counter {  
  3.      public Counter() {  
  4.      }  
  5.      public int count(ArrayList items) {  
  6.           int results = 0;  
  7.           for(Object curItem : items) {  
  8.                results ++;  
  9.           }  
  10.           return results;  
  11.      }  
  12. }   
  如你所见,上面的例子十分简单,但它阐明了要点。当你想要测试count方法时,你会针对count方法本身如何工作去写测试代码。你不会去测试ArrayList是否正常工作,因为你默认它已经被测过并且工作正常。你唯一的目标就是测试对ArrayList的使用。
  模拟对象的概念就是我们想要创建一个可以替代实际对象的对象。这个模拟对象要可以通过特定参数调用特定的方法,并且能返回预期结果。

 

 

模拟有哪些关键点?

 

 

  在谈到模拟时,你只需关心三样东西:设置测试数据,设定预期结果,验证结果。一些单元测试方案根本就不涉及这些,有的只涉及设置测试数据,有的只涉及设定预期结果和验证。

 

Stubbing (桩)

  Stubbing就是告诉fake当与之交互时执行何种行为过程。通常它可以用来提供那些测试所需的公共属性(像getters和setters)和公共方法。

  当谈到stubbing方法,通常你有一系列的选择。或许你希望返回一个特殊的值,抛出一个错误或者触发一个事件,此外,你可能希望指出方法被调用时的不同行为(即通过传递匹配的类型或者参数给方法)。

  这咋一听起来工作量很大,但通常并非这样。许多mocking框架的一个重要功能就是你不需要提供stub 的实体方法,也不用在执行测试期间stub那些未被调用的方法或者未使用的属性。

设置预期

  Fake的一个关键的特性就是当你用它进行模拟测试时你能够告诉它你预期的结果。例如,你可以要求一个特定的函数被准确的调用3次,或不被调用,或调用至少两次但不超过5次,或者需要满足特定类型的参数、特定值和以上任意的组合的调用。可能性是无穷的。

  通过设定预期结果告诉fake你期望发生的事情。因为它是一个模拟测试,所以实际上什么也没发生。但是,对于被测试的类来说,它并无法区分这种情况。所以fake能够调用函数并让它做它该做的。

  值得注意的是,大多数模拟框架除了可以创建接口的模拟测试外,还可以创建公有类的模拟测试。

验证预期结果

  设置预期和验证预期是同时进行的。设置预期在调用测试类的函数之前完成,验证预期则在它之后。所以,首先你设定好预期结果,然后去验证你的预期结果是否正确。

  在一个单元测试中,如果你设定的预期没有得到满足,那么这个单元测试就是失败了。例如,你设置预期结果是 ILoginService.login函数必须用特定的用户名和密码被调用一次,但是在测试中它并没有被调用,这个fake没被验证,所以测试失败。


模拟的好处是什么? 

   提前创建测试; TDD(测试驱动开发)  

  这是个最大的好处吧。如果你创建了一个Mock那么你就可以在service接口创建之前写Service Tests了,这样你就能在开发过程中把测试添加到你的自动化测试环境中了。换句话说,模拟使你能够使用测试驱动开发。 

   团队可以并行工作 

  这类似于上面的那点;为不存在的代码创建测试。但前面讲的是开发人员编写测试程序,这里说的是测试团队来创建。当还没有任何东西要测的时候测试团队如何来创建测试呢?模拟并针对模拟测试!这意味着当service借口需要测试时,实际上QA团队已经有了一套完整的测试组件;没有出现一个团队等待另一个团队完成的情况。这使得模拟的效益型尤为突出了。 

   你可以创建一个验证或者演示程序。 

  由于Mocks非常高效,Mocks可以用来创建一个概念证明,作为一个示意图,或者作为一个你正考虑构建项目的演示程序。这为你决定项目接下来是否要进行提供了有力的基础,但最重要的还是提供了实际的设计决策。 

  为无法访问的资源编写测试 

  这个好处不属于实际效益的一种,而是作为一个必要时的“救生圈”。有没有遇到这样的情况?当你想要测试一个service接口,但service需要经过防火墙访问,防火墙不能为你打开或者你需要认证才能访问。遇到这样情况时,你可以在你能访问的地方使用MockService替代,这就是一个“救生圈”功能。 

  Mock 可以分发给用户 

      在有些情况下,某种原因你需要允许一些外部来源访问你的测试系统,像合作伙伴或者客户。这些原因导致别人也可以访问你的敏感信息,而你或许只是想允许访问部分测试环境。在这种情况下,如何向合作伙伴或者客户提供一个测试系统来开发或者做测试呢?最简单的就是提供一个mock,无论是来自于你的网络或者客户的网络。soapUI mock非常容易配置,他可以运行在soapUI或者作为一个war包发布到你的java服务器里面。 

  隔离系统 

      有时,你希望在没有系统其他部分的影响下测试系统单独的一部分。由于其他系统部分会给测试数据造成干扰,影响根据数据收集得到的测试结论。使用mock你可以移除掉除了需要测试部分的系统依赖的模拟。当隔离这些mocks后,mocks就变得非常简单可靠,快速可预见。这为你提供了一个移除了随机行为,有重复模式并且可以监控特殊系统的测试环境。 


Mockito 框架

 

  Mockito 是一个基于MIT协议的开源java测试框架。 

 

Mockito是一个强大的基于MIT协议的用于Java开发的模拟测试框架。其他还有 EasyMock,JMock,MockCreator,Mockrunner,MockMaker 及 PowerMock。

 

  Mockito区别于其他模拟框架的地方主要是允许开发者在没有建立“预期”时验证被测系统的行为。对mock对象的一个批评是测试代码与被测系统高度耦合,由于Mockito试图通过移除“期望规范”来去除expect-run-verify模式(期望--运行--验证模式),因此使耦合度降低到最低。这样的突出特性简化了测试代码,使它更容易阅读和修改了。

你可以验证交互:

[java] view plain copy
 
  1. // 模拟的创建,对接口进行模拟  
  2. List mockedList = mock(List.class);  
  3. // 使用模拟对象  
  4. mockedList.add("one");  
  5. mockedList.clear();  
  6. // 选择性地和显式地验证  
  7. verify(mockedList).add("one");  
  8. verify(mockedList).clear();    

或者存根方法调用:

[java] view plain copy
 
  1. // 你不仅可以模拟接口,任何具体类都行  
  2. LinkedList mockedList = mock(LinkedList.class);  
  3. // 执行前准备测试数据  
  4. when(mockedList.get(0)).thenReturn("first");  
  5. // 接着打印"first"  
  6. System.out.println(mockedList.get(0));  
  7. // 因为get(999)未对准备数据,所以下面将打印"null".  
  8. System.out.println(mockedList.get(999));   

一个使用Mockito框架的简单Java代码示例



图2 不使用Mock框架


图3 使用Mockito框架

步骤 1:  在IDE中创建一个普通的Java项目 

  在Eclipse、NetBeans或IntelliJ IDEA中创建一个普通的Java项目。

步骤 2:  添加java源码  

  类Person.java:

[java] view plain copy
 
  1. package mockitodemo;  
  2.   
  3. public class Person  
  4. {  
  5.     private final Integer personID;  
  6.     private final String personName;  
  7.     public Person( Integer personID, String personName )  
  8.     {  
  9.         this.personID = personID;  
  10.         this.personName = personName;  
  11.     }  
  12.     public Integer getPersonID()  
  13.     {  
  14.         return personID;  
  15.     }  
  16.     public String getPersonName()  
  17.     {  
  18.         return personName;  
  19.     }  
  20. }   
  接口PersonDAO.java
[java] view plain copy
 
  1. package mockitodemo;  
  2.   
  3. public interface PersonDao  
  4. {  
  5.     public Person fetchPerson( Integer personID );  
  6.     public void update( Person person );  
  7. }   
  类PersonService.java
[java] view plain copy
 
  1. package mockitodemo;  
  2.   
  3. public class PersonService  
  4. {  
  5.     private final PersonDao personDao;  
  6.     public PersonService( PersonDao personDao )  
  7.     {  
  8.         this.personDao = personDao;  
  9.     }  
  10.     public boolean update( Integer personId, String name )  
  11.     {  
  12.         Person person = personDao.fetchPerson( personId );  
  13.         if( person != null )  
  14.         {  
  15.             Person updatedPerson = new Person( person.getPersonID(), name );  
  16.             personDao.update( updatedPerson );  
  17.             return true;  
  18.         }  
  19.         else  
  20.         {  
  21.             return false;  
  22.         }  
  23.     }  
  24. }   

步骤 3:  添加单元测试类.  

  接下来为类PersonService.java创建单元测试用例。我们使用JUnit 4.x和Mockito 1.9.5。可以设计测试用例类PersionServiceTest.java为如下,代码中有详细注释说明:

[java] view plain copy
 
  1. package mockitodemo;  
  2.   
  3. import org.junit.After;  
  4. import org.junit.AfterClass;  
  5. import org.junit.Before;  
  6. import org.junit.BeforeClass;  
  7. import org.junit.Test;  
  8. import static org.junit.Assert.*;  
  9. import org.mockito.Mock;  
  10. import org.mockito.MockitoAnnotations;  
  11. import org.mockito.ArgumentCaptor;  
  12. import static org.mockito.Mockito.*;  
  13.   
  14. /** 
  15.  * PersonService的单元测试用例 
  16.  * 
  17.  * @author jackzhou 
  18.  */  
  19. public class PersonServiceTest {  
  20.   
  21.     @Mock  
  22.     private PersonDao personDAO;  // 模拟对象  
  23.     private PersonService personService;  // 被测类  
  24.   
  25.     public PersonServiceTest() {  
  26.     }  
  27.   
  28.     @BeforeClass  
  29.     public static void setUpClass() {  
  30.     }  
  31.   
  32.     @AfterClass  
  33.     public static void tearDownClass() {  
  34.     }  
  35.   
  36.     // 在@Test标注的测试方法之前运行  
  37.     @Before  
  38.     public void setUp() throws Exception {  
  39.         // 初始化测试用例类中由Mockito的注解标注的所有模拟对象  
  40.         MockitoAnnotations.initMocks(this);  
  41.         // 用模拟对象创建被测类对象  
  42.         personService = new PersonService(personDAO);  
  43.     }  
  44.   
  45.     @After  
  46.     public void tearDown() {  
  47.     }  
  48.   
  49.     @Test  
  50.     public void shouldUpdatePersonName() {  
  51.         Person person = new Person(1"Phillip");  
  52.         // 设置模拟对象的返回预期值  
  53.         when(personDAO.fetchPerson(1)).thenReturn(person);  
  54.         // 执行测试  
  55.         boolean updated = personService.update(1"David");  
  56.         // 验证更新是否成功  
  57.         assertTrue(updated);  
  58.         // 验证模拟对象的fetchPerson(1)方法是否被调用了一次  
  59.         verify(personDAO).fetchPerson(1);  
  60.         // 得到一个抓取器  
  61.         ArgumentCaptor<Person> personCaptor = ArgumentCaptor.forClass(Person.class);  
  62.         // 验证模拟对象的update()是否被调用一次,并抓取调用时传入的参数值  
  63.         verify(personDAO).update(personCaptor.capture());  
  64.         // 获取抓取到的参数值  
  65.         Person updatePerson = personCaptor.getValue();  
  66.         // 验证调用时的参数值  
  67.         assertEquals("David", updatePerson.getPersonName());  
  68.         // asserts that during the test, there are no other calls to the mock object.  
  69.         // 检查模拟对象上是否还有未验证的交互  
  70.         verifyNoMoreInteractions(personDAO);  
  71.     }  
  72.   
  73.     @Test  
  74.     public void shouldNotUpdateIfPersonNotFound() {  
  75.         // 设置模拟对象的返回预期值  
  76.         when(personDAO.fetchPerson(1)).thenReturn(null);  
  77.         // 执行测试  
  78.         boolean updated = personService.update(1"David");  
  79.         // 验证更新是否失败  
  80.         assertFalse(updated);  
  81.         // 验证模拟对象的fetchPerson(1)方法是否被调用了一次  
  82.         verify(personDAO).fetchPerson(1);  
  83.         // 验证模拟对象是否没有发生任何交互  
  84.         verifyZeroInteractions(personDAO);  
  85.         // 检查模拟对象上是否还有未验证的交互  
  86.         verifyNoMoreInteractions(personDAO);  
  87.     }      
  88.   
  89.     /** 
  90.      * Test of update method, of class PersonService. 
  91.      */  
  92.     @Test  
  93.     public void testUpdate() {  
  94.         System.out.println("update");  
  95.         Integer personId = null;  
  96.         String name = "Phillip";  
  97.         PersonService instance = new PersonService(new PersonDao() {  
  98.   
  99.             @Override  
  100.             public Person fetchPerson(Integer personID) {  
  101.                 System.out.println("Not supported yet.");  
  102.                 return null;  
  103.             }  
  104.   
  105.             @Override  
  106.             public void update(Person person) {  
  107.                 System.out.println("Not supported yet.");  
  108.             }  
  109.         });  
  110.         boolean expResult = false;  
  111.         boolean result = instance.update(personId, name);  
  112.         assertEquals(expResult, result);  
  113.         // TODO review the generated test code and remove the default call to fail.  
  114.         fail("The test case is a prototype.");  
  115.     }  
  116. }  
  这里setUpClass()、tearDownClass()、setUp()、tearDown()称为测试夹具(Fixture),就是测试运行程序(test runner)在运行测试方法之前进行初始化、或之后进行回收资源的工作。JUnit 4之前是通过setUp、tearDown方法完成。在JUnit 4中,仍然可以在每个测试方法运行之前初始化字段和配置环境,当然也是通过注解完成。在JUnit 4中,通过@Before标注setUp方法;@After标注tearDown方法。在一个测试类中,甚至可以使用多个@Before来注解多个方法,这些方法都是在每个测试之前运行。说明一点,一个测试用例类可以包含多个打上@Test注解的测试方法,在运行时,每个测试方法都对应一个测试用例类的实例。@Before是在每个测试方法运行前均初始化一次,同理@Ater是在每个测试方法运行完毕后均执行一次。也就是说,经这两个注解的初始化和注销,可以保证各个测试之间的独立性而互不干扰,它的缺点是效率低。另外,不需要在超类中显式调用初始化和清除方法,只要它们不被覆盖,测试运行程序将根据需要自动调用这些方法。超类中的@Before方法在子类的@Before方法之前调用(与构造函数调用顺序一致),@After方法是子类在超类之前运行。
  这里shouldUpdatePersonName()、shouldNotUpdateIfPersonNotFound()和testUpdate()都是测试PersonService的update()方法,它依赖于PersonDao接口。前两者使用了模拟测试。testUpdate()则没有使用模拟测试。下面是测试结果:

图4 测试结果点击打开链接

  可以看出,使用模拟测试的两个测试成功了,没有使用模拟测试的testUpdate()失败。对于模拟测试,在测试用例类中要先声明依赖的各个模拟对象,在setUp()中用MockitoAnnotations.initMocks()初始化所有模拟对象。在进行模拟测试时,要先设置模拟对象上方法的返回预期值,执行测试时会调用模拟对象上的方法,因此要验证这些方法是否被调用,并且传入的参数值是否符合预期。对于testUpdate()测试,我们需要自己创建测试PersonService.update()所需的所有PersonDao数据,因为我们只知道公开的PersonDao接口,其具体实现类(比如从数据库中拿真实的数据,或写入到数据库中)可能由另一个团队在负责,以适配不同的数据库系统。这样的依赖关系无疑使单元测试比较麻烦,而要拿真正PersonDao实现来进行测试,那也应该是后期集成测试的任务,把不同的组件集成到一起在真实环境中测试。有了模拟测试框架,就可以最大限度地降低单元测试时的依赖耦合性。

关注点 

 + Mock框架是什么.

 + 为什么要在测试中使用Mockito

参考

http://java.dzone.com/articles/the-concept-mocking  

http://en.wikipedia.org/wiki/Mockito 

http://code.google.com/p/mockito

http://blog.csdn.net/zhoudaxia/article/details/33056093

http://mp.weixin.qq.com/s?__biz=MjM5NzMyMjAwMA==&mid=401219438&idx=2&sn=b1be5b58c5740e5475bb28bae6fbd8f8&3rd=MzA3MDU4NTYzMw==&scene=6#rd

分享到:
评论

相关推荐

    java自动化测试框架

    本文将深入探讨Java自动化测试框架的核心概念、常用工具以及如何构建一个基本的自动化测试流程。 首先,我们要理解自动化测试的重要性。在软件开发中,手动测试虽然直观,但耗时且容易出错。自动化测试通过编写可...

    Mockito:用 Java 编写的最流行的单元测试模拟框架-开源

    可以从 Central Repository 和 Bintray ... Mockito 是一个很好用的模拟框架。 它让您可以使用干净简单的 API 编写漂亮的测试。 Mockito 不会让您感到宿醉,因为这些测试非常具有可读性,并且会产生干净的验证错误。

    spring集成TestNG与Mockito框架单元测试方法

    Mockito是一个流行的Java模拟框架,它允许你在测试中隔离依赖,专注于被测试代码的行为。在你的项目中引入Mockito依赖: ```xml &lt;groupId&gt;org.mockito &lt;artifactId&gt;mockito-core &lt;version&gt;4.2.0 &lt;scope&gt;test ...

    Mockito:最受欢迎的用Java编写的用于单元测试的Mocking框架-开源

    可从中央存储库和Bintray获得Mockito下载以及... Mockito是一个非常不错的模拟框架。 它使您可以使用干净简单的API编写漂亮的测试。 Mockito不会给您带来麻烦,因为这些测试的可读性很强,并且会产生清晰的验证错误。

    Java单元测试框架Demo

    Mockito是一个强大的模拟框架,它允许我们在测试中替换掉依赖的外部组件,创建“mock对象”来隔离被测试的代码。例如,如果`MyClass`依赖于`AnotherClass`,我们可以在测试`MyClass`时使用Mockito创建`AnotherClass`...

    Java-Junit-Mockito

    Green Bar和assertEquals步骤04:重构您的第一个JUnit测试步骤05:第二个JUnit示例assertTrue和assertFalse步骤06:@Before @After步骤07:@BeforeClass @AfterClass步骤08:比较JUnit测试中的数组步骤09:测试...

    mockito源码 供java同学学习,用于在自动化单元测试中创建测试对象,为TDD或BDD提供支持

    Mockito是一个为Java开发的强大的模拟测试框架。其主要功能是创建和配置Mock对象,以简化存在外部依赖的类的测试。通过使用Mockito,我们可以虚拟出一个外部依赖,这样可以降低测试组件之间的耦合度,使得我们能更...

    javaweb开发之道源码-mockito:用Java编写的最流行的单元测试Mocking框架

    模拟框架 当前版本是 3.x 还在 Mockito 1.x 上? 在 Mockito 2 中看到! 不会引入任何破坏性的 API 更改,但现在 Mockito 2 需要 Java 8 而不是 Java 6。 企业版 Mockito 作为 Tidelift 订阅的一部分提供 org....

    java 测试

    “工具”可能指的是用于辅助测试的各种Java工具和库,如Mockito用于模拟对象,JMockit用于无侵入式模拟,或者使用Apache Maven或Gradle构建工具中的插件进行自动化测试和报告生成。此外,持续集成工具如Jenkins、...

    Mockito:休息测试

    Mockito 是一个流行的 Java 单元测试框架,它允许开发者模拟和验证对象的行为,以便在测试过程中隔离被测试代码。休息测试(Rest Testing)通常指的是针对 RESTful Web 服务进行的功能测试,确保 API 能够按照预期...

    java测试必备工具【用例】

    - Mockito:一个模拟对象框架,允许在测试中隔离依赖,只关注要测试的代码部分。 - Spring Boot Test:针对Spring Boot应用的集成测试库,可以轻松地测试服务、控制器等。 3. 自动化测试工具: - Selenium:用于...

    Mockito:用 Java 编写的最流行的单元测试 Mocking 框架-开源

    可以从 Central Repository 和 Bintray ... Mockito 是一个很好用的模拟框架。 它让您可以使用干净简单的 API 编写漂亮的测试。 Mockito 不会让您感到宿醉,因为这些测试非常具有可读性,并且会产生干净的验证错误。

    mockito-1.9.5.zip

    Mockito 是一个流行的Java单元测试框架,用于模拟对象行为,使得测试更为简洁和可控。JUnit则是最常用的Java单元测试库,它提供了一种结构化的方式来编写和运行测试用例。当我们进行单元测试时,Mockito 和 JUnit ...

    java UT mockito配套代码

    Java单元测试(Unit Testing)是软件开发中的一个重要环节,它用于验证代码的各个独立部分是否按预期工作。Mockito是一个流行的Java单元测试框架,它允许开发者创建和配置模拟对象(mock objects),以便在测试中...

    最流行的用 Java 编写的单元测试 Mocking 框架.zip

    最流行的 Java 模拟框架 当前版本是 5.x仍在使用 Mockito 1.x?查看Mockito 2 中的新功能! Mockito 3不会引入任何重大 API 更改,但现在 Mockito 2 需要 Java 8 而不是 Java 6。Mockito 4删除了已弃用的 API。...

    java开发架包

    11. **Mockito**:Mockito是一个Java单元测试框架,它允许开发者模拟对象的行为,以便在测试中隔离被测代码。 12. **Jackson / Gson**:这两个库用于JSON序列化和反序列化,将Java对象转换为JSON字符串,反之亦然,...

    java自动化测试

    Java自动化测试是软件开发过程中一个重要的环节,它利用编程语言如Java来实现测试脚本,以提高测试效率,减少人工错误,并确保软件质量。在Java自动化测试中,我们通常会用到一系列工具和技术,包括单元测试、集成...

    基于Mockito和PowerMockito的Java单元测试指南.zip

    本项目是一个专注于Java单元测试的指南,特别是使用Mockito和PowerMockito框架进行模拟测试。项目涵盖了从基础的Mockito使用到高级的PowerMockito功能,帮助开发者掌握如何在不同场景下进行有效的单元测试。 项目...

    Mastering Unit Testing Using Mockito and JUnit.pdf

    其次,Mockito是一个流行的Java模拟框架,它允许我们在测试中创建和配置模拟对象。这些模拟对象可以代替真实依赖,以便我们专注于测试单个组件的行为,而无需考虑整个系统。Mockito的核心概念包括mock、spy、when和...

Global site tag (gtag.js) - Google Analytics