from:http://www.blogjava.net/supercrsky/articles/162766.html

Mock 对象能够模拟领域对象的部分行为,并且能够检验运行结果是否和预期的一致。领域类将通过与 Mock 对象的交互,来获得一个独立的测试环境(引自《 精通 Spring——Java 轻量级架构开发实践 》。

在模仿对象中,我们定义了四个概念:

1 )目标对象:正在测试的对象

2 )合作者对象:由目标对象创建或获取的对象

3 )模仿对象:遵循模仿对象模式的合作者的子类(或实现)

4 )特殊化对象:覆盖创建方法以返回模仿对象而不是合作者的目标的子类

一般来说,应用模仿对象的过程如下:

1 )创建模仿对象的实例

2 )设置模仿对象中的状态和期望值

3 )将模仿对象作为参数来调用域代码

4 )验证模仿对象中的一致性

那么,我们应该如何以及在哪里使用 Mock 对象呢?一般来说,对于目标对象中的合作者对象,在测试时如果其状态或行为的实现严重地依赖外部资源(比如数据持久化中的 DAO ,比如负责发送电子邮件的类),或者团队并行开发时,目标对象的合作者对象并没有实现(比如 J2EE 中,横向分工时,负责 Action 的调用 Service ,负责 Service 调用 DAO 时,相应的 Service 及 DAO 没有实现),这时我们就需要模仿这些类。其实,在做J2EE 时,传统的 N 层架构中,我们都是面向接口编程的,我们定义了 DAO 接口,我们定义了 Service 接口,这样做的优点就是我们在测试时可以构造实现接口的 Mock 类。这里不得不提依赖注入,通过依赖注入,我们才能在测试时 set Mock 对象。这也说明,为了方便测试,我们不得不一步一步 重构代码,而模式就在重构中自然地产生了。

对于 Mock 对象,我们可以根据 合作者接口(或者是类) 实现具体的 Mock 类,这样的 Mock 类实际上是Stub 。有些情况下, Stub 是必要的。但对于诸如 DAO 、 Service ,我们只关心在给定参数的情况下,调用的方法能够返回预期的值,我们根本不关心其内部实现,这时候如果使用 Stub 的话就会产生不必要的代码。我们需要的就是能 动态地生成 Mock 对象而不需要编写它们的工具, EasyMock 就是这样的工具。

EasyMock 是一个 Mock 对象的类库,现在的版本是 2.0 ,这个版本只支持 Mock 接口,如果需要 Mock 类,需要下载它的扩展包。

下面通过一个具体的例子说明一下 Mock 对象的使用,我写的例子就是测试 Service 类中的一个方法, Mock的对象是 DAO 。

首先是一个简单的实体 bean Account:

 package  easymocktest.domain;

 import  org.apache.commons.lang.builder.ToStringBuilder;
 import  org.apache.commons.lang.builder.ToStringStyle;

 public   class  Account  {

     private  Long id;
     private  String name;
     private  String pwd;
    
     public  Long getId()  {
         return  id;
    } 
      public   void  setId(Long id)  {
         this .id  =  id;
    } 
      public  String getName()  {
         return  name;
    } 
      public   void  setName(String name)  {
         this .name  =  name;
    } 
      public  String getPwd()  {
         return  pwd;
    } 
      public   void  setPwd(String pwd)  {
         this .pwd  =  pwd;
    } 
      /** 
     *  @see  java.lang.Object#toString()
      */ 
      public  String toString()  {
         return   new  ToStringBuilder( this , ToStringStyle.MULTI_LINE_STYLE)
                .append( " name " ,  this .name).append( " pwd " ,  this .pwd).append( " id " ,
                         this .id).toString();
    } 
    

一个只含一个方法的 DAO AccountDAO:

 package  easymocktest.dao;

 import  easymocktest.domain. * ;
 public   interface  AccountDAO  {
     public  Account getByNameAndPwd(String name,String pwd);

一个同样只含一个方法的 AccountService 接口:

 package  easymocktest.service;

 import  easymocktest.domain. * ;
 public   interface  AccountService  {
     public  Account getAccount(String name,String pwd);

与 AccountService 相应的实现类:

 package  easymocktest.service.impl;

 import  easymocktest.service.AccountService;
 import  easymocktest.dao. * ;
 import  easymocktest.domain.Account;
 public   class  AccountServiceImpl  implements  AccountService  {

     private  AccountDAO accountDAO;

     public  AccountDAO getAccountDAO()  {
         return  accountDAO;
    } 
 
      public   void  setAccountDAO(AccountDAO accountDAO)  {
         this .accountDAO  =  accountDAO;
    } 
 
      public  Account getAccount(String name, String pwd)  {
         return   this .accountDAO.getByNameAndPwd(name, pwd);
    } 
 

这里我没有实现 AccountDAO 接口,对于 Mock 测试来说,这也是不需要的。下面就是 AccountServiceImpl 的测试类AccountServiceTest :

 package  easymocktest.service;

 import  junit.framework. * ;
 import  easymocktest.dao. * ;
 import  easymocktest.domain. * ;
 import  easymocktest.service.impl. * ;
 import   static  org.easymock.EasyMock.createMock;
 import   static  org.easymock.EasyMock.replay;
 import   static  org.easymock.EasyMock.reset;
 import   static  org.easymock.EasyMock.verify;
 import   static  org.easymock.EasyMock.expect;
 public   class  AccountServiceTest  extends  TestCase {

     private  AccountDAO accountDAOMock;
     private  AccountServiceImpl accountService;
    
    @Override
     protected   void  setUp()  throws  Exception  {
        accountDAOMock  =  createMock(AccountDAO. class );
        accountService  =   new  AccountServiceImpl();
        accountService.setAccountDAO(accountDAOMock);
    } 
    
     public   void  testGetAccount() {
        String name  =   " kafka " ;
        String pwd  =   " 0102 " ;
        Account a  =   new  Account();
        a.setName(name);
        a.setPwd(pwd);
        a.setId( new  Long( 10 ));
        reset(accountDAOMock); // (a) 
         expect(accountDAOMock.getByNameAndPwd(name, pwd)).andReturn(a); // (b) 
         replay(accountDAOMock); // (c) 
         Account b  =  accountService.getAccount(name, pwd);
        assertEquals(a, b);
        verify(accountDAOMock); // (d) 
     } 

下面简要的说明一下 Mock 对象的工作过程:

1 )在 setUp() 中,通过 “accountDAOMock = createMock(AccountDAO.class);” (这里使用了 java5 中的静态导入),创建AccountDAO 的 Mock 对象,由于 EasyMock 采用了范型技术,故创建的 Mock 对象不需要强制类型转换。然后通过“accountService.setAccountDAO(accountDAOMock);” 设置目标对象的合作者对象。

2 )对于测试方法 “testGetAccount()” , (a) 处的 reset() 方法是将 Mock 对象复位,也就是重新设置 Mock 对象的状态和行为。由于此处是第一次调用 Mock 对象,可以不必使用 reset() 方法。

3 ) (b) 处 expect() 是录制 Mock 对象方法的调用,其参数就是 Mock 对象的方法,其中如果调用的方法有返回值,要通过andReturn() 方法设置预期的返回值。

4 ) (c) 处的 replay() 是结束录制过程。 在调用 replay() 方法之前的状态, EashMock 称之为 “record 状态 ” 。该状态下, Mock 对象不具备行为(即模拟接口的实现),它仅仅记录方法的调用。在调用 replay() 后,它才以 Mock 对象预期的行为进行工作,检查预期的方法调用是否真的完成。

5 ) (d) 处的 verify() 是用于在录制和回放两个步骤完成之后进行预期和实际结果的检查。这里就是检查 accountDAOMock 是否如预期一样调用了 getByNameAndPwd 方法。

对于上面的举例,它可能并不具有实际的价值,这里我只想抛砖引玉。在 N 层架构的 Java 程序中, Mock对象在单元测试中正发挥着越来越重要的作用。我现在看到的是,在 Service 层与 Web 层, Mock 对象能很好的被应用。有人觉得在 Persistence 层也应该使用 Mock 对象,但就像我们所知道的,在使用 Hibernate 、Ibatis 等 ORM 工具的情况下,我们的 Persistence 层的测试主要测试的就是那些配置文件、查询语句等(实际上是集成测试),如果还 Mock 的话,就失去了测试的意义。

对于 Mock 的更多的信息,你可以访问Mock Objects,在本文写作的过程中,参考了使用模仿对象进行单元测试

等文章,一并感谢。

EasyMock使用手记的更多相关文章

  1. uni-app官方教程学习手记

    本人微信公众号:前端修炼之路,欢迎关注 背景介绍 大概在今年的十月份左右,我了解到Dcloud推出了uni-app.当时下载了一个Hbuilder X,下载了官方提供的hello示例教程.经过一番努力 ...

  2. Linux.NET实战手记—自己动手改泥鳅(上)

    各位读者大家好,不知各位读者有否阅读在下的前一个系列<Linux.NET 学习手记>,在前一个系列中,我们从Linux中Mono的编译安装开始,到Jexus服务器的介绍,以及如何在Linu ...

  3. Linux.NET学习手记(7)

    前一篇中,我们简单的讲述了下如何在Linux.NET中部署第一个ASP.NET MVC 5.0的程序.而目前微软已经提出OWIN并致力于发展VNext,接下来系列中,我们将会向OWIN方向转战. 早在 ...

  4. Linux.NET学习手记(8)

    上一回合中,我们讲解了Linux.NET面对OWIN需要做出的准备,以及介绍了如何将两个支持OWIN协议的框架:SignalR以及NancyFX以OwinHost的方式部署到Linux.NET当中.这 ...

  5. 关于《Linux.NET学习手记(8)》的补充说明

    早前的一两天<Linux.NET学习手记(8)>发布了,这一篇主要是讲述OWIN框架与OwinHost之间如何根据OWIN协议进行通信构成一套完整的系统.文中我们还直接学习如何直接操作OW ...

  6. U3D DrawCall优化手记

    在最近,使用U3D开发的游戏核心部分功能即将完成,中间由于各种历史原因,导致项目存在比较大的问题,这些问题在最后,恐怕只能通过一次彻底的重构来解决 现在的游戏跑起来会有接近130-170个左右的Dra ...

  7. 信息系统实践手记5-CACHE设计一例

    说明:信息系统实践手记系列是系笔者在平时研发中先后遇到的大小的问题,也许朴实和细微,但往往却是经常遇到的问题.笔者对其中比较典型的加以收集,描述,归纳和分享. 摘要:此文描述了笔者接触过的部分信息系统 ...

  8. 信息系统实践手记6-JS调用Flex的性能问题一例

    说明:信息系统实践手记系列是系笔者在平时研发中先后遇到的大小的问题,也许朴实和细微,但往往却是经常遇到的问题.笔者对其中比较典型的加以收集,描述,归纳和分享. 摘要:此文描述了笔者接触过的部分信息系统 ...

  9. 测试--easymock的使用

    使用场景:对于调用其它类中的方法,但是还没有编写完,使用easymock进行单元测试,它提供这些没有编写完的代码期待的默认值. 使用步骤: step1: pom引入: <dependency&g ...

随机推荐

  1. SQL Server 2008 序列号

    SQL Server 2008 序列号:Developer: PTTFM-X467G-P7RH2-3Q6CG-4DMYBEnterprise:   JD8Y6-HQG69-P9H84-XDTPG-34 ...

  2. 【转】eclipse怎么设置字体大小

    原文网址:http://jingyan.baidu.com/article/f96699bb9442f3894e3c1b15.html 1. 打开eclipse,找到window 2.  点击后在下拉 ...

  3. OpenGL学习之路(二)

    1 引子 在上一篇读书笔记中,我们对书本中给出的例子进行详细的分析.首先是搭出一个框架:然后填充初始化函数,在初始化函数中向OpenGL提供顶点信息(缓冲区对象)和顶点属性信息(顶点数组对象),并启用 ...

  4. iOS设计模式——Category

    什么是Category Category模式用于向已经存在的类添加方法从而达到扩展已有类的目的,在很多情形下Category也是比创建子类更优的选择.新添加的方法同样也会被被扩展的类的所有子类自动继承 ...

  5. 可接受多个值的文件上传字段HTML5新特性

    <input type="file" id="input"  multiple="multiple"> 主要是多了个multip ...

  6. Golang做的验证码(2)

    前面一篇文章介绍了2个用Golang做的验证码 http://www.cnblogs.com/ghj1976/p/3392847.html  这里再补充几个: 1.在GAE上使用的Google的验证码 ...

  7. Mountain Road

    题意: n个车,过一条路,有不同的方向,路上不允许同时有两个方向的车,给出每个车的起始时间,方向,和经过路花费的时间,车最小间隔10个时间,求最后一个车通过路的最早的时间. 分析: dp[i][j][ ...

  8. HDU5828 Rikka with Sequence 线段树

    分析:这个题和bc round 73应该是差不多的题,当时是zimpha巨出的,那个是取phi,这个是开根 吐槽:赛场上写的时候直接维护数值相同的区间,然后1A,结果赛后糖教一组数据给hack了,仰慕 ...

  9. 解决Asp.net Mvc返回JsonResult中DateTime类型数据格式的问题

    问题背景: 在使用asp.net mvc 结合jquery esayui做一个系统,但是在使用使用this.json方法直接返回一个json对象,在列表中显示时发现datetime类型的数据在转为字符 ...

  10. js获取字符串最后一个字符代码

    方法一:运用String对象下的charAt方法 charAt() 方法可返回指定位置的字符. 代码如下 复制代码 str.charAt(str.length – 1) 请注意,JavaScript ...