· [Java论坛][安全论坛][数据库论坛][操作系统论坛]
· [专题] IBM Rational高峰论坛 Windows 7十大功能
· [专题] 史上就危险7月 微软 BizSpark 计划介绍
· [IT技术周刊][IT资源下载专区][病毒求助专区]
· [热点] 我也能做CTO_赛迪连载 赛迪七夕特别行动
· [热点] Chrome 4.0采用V8引擎 Java开发编程规范

使用 MOCK 对象进行单元测试的实例讲解 (1)

发布时间:2007.08.10 06:19     来源:赛迪网    作者:dxaw

1.出了什么问题?

单元测试的目标是一次只验证一个方法,小步的前进,细粒度的测试,但是假如某个方法依赖于其他一些难以操控的东东,比如说网络连接,数据库连接,或者是Servlet容器,那么我们该怎么办呢?

要是你的测试依赖于系统的其他部分,甚至是系统的多个其他部分呢?在这种情况下,倘若不小心,你最终可能会发现自己几乎初始化了系统的每个组件,而这只是为了给一个测试创造足够的运行环境让它们可以运行起来。忙乎了大半天,看上去我们好像有点违背了测试的初衷了。这样不仅仅消耗时间,还给测试过程引入了大量的耦合因素,比如说,可能有人兴致冲冲地改变了一个接口或者数据库的一张表,突然,你那卑微的单元测试的神秘的挂掉了。在这种情况发生几次之后,即使是最有耐心的开发者也会泄气,甚至最终放弃所有的测试,那样的话后果就不能想像了。

再让我们看一个更加具体的情况:在实际的面向对象软件设计中,我们经常会碰到这样的情况,我们在对现实对象进行构建之后,对象之间是通过一系列的接口来实现。这在面向对象设计里是最自然不过的事情了,但是随着软件测试需求的发展,这会产生一些小问题。举个例子,用户A现在拿到一个用户B提供的接口,他根据这个接口实现了自己的需求,但是用户A编译自己的代码后,想简单模拟测试一下,怎么办呢?这点也是很现实的一个问题。我们是否可以针对这个接口来简单实现一个代理类,来测试模拟,期望代码生成自己的结果呢?

幸运的是,有一种测试模式可以帮助我们:mock对象。Mock对象也就是真实对象在调试期的替代品。

2.现在需要Mock对象吗?

关于什么时候需要Mock对象,Tim Mackinnon给我们了一些建议:

----- 真实对象具有不可确定的行为(产生不可预测的结果,如股票的行情)

----- 真实对象很难被创建(比如具体的web容器)

----- 真实对象的某些行为很难触发(比如网络错误)

----- 真实情况令程序的运行速度很慢

----- 真实对象有用户界面

----- 测试需要询问真实对象它是如何被调用的(比如测试可能需要验证某个回调函数是否被调用了)

----- 真实对象实际上并不存在(当需要和其他开发小组,或者新的硬件系统打交道的时候,这是一个普遍的问题)

3.如何实现Mock对象?

使用mock对象进行测试的时候,我们总共需要3个步骤,分别是:

----- 使用一个接口来描述这个对象

----- 为产品代码实现这个接口

----- 以测试为目的,在mock对象中实现这个接口

在此我们又一次看到了针对接口编程的重要性了,因为被测试的代码只会通过接口来引用对象,所以它完全可以不知道它引用的究竟是真实的对象还是mock对象,下面看一个实际的例子:一个闹钟根据时间来进行提醒服务,如果过了下午5点钟就播放音频文件提醒大家下班了,如果我们要利用真实的对象来测试的话就只能苦苦等到下午五点,然后把耳朵放在音箱旁...我们可不想这么笨,我们应该利用mock对象来进行测试,这样我们就可以模拟控制时间了,而不用苦苦等待时钟转到下午5点钟了。下面是代码:

public interface Environmental {    
private boolean playedWav = false;    
public long getTime();    
public void playWavFile(String fileName);    
public boolean wavWasPlayed();    
public void resetWav();    
}

真实的实现代码:

public class SystemEnvironment implements Environmental {    
public long getTime() {    
return System.currentTimeMillis();    
      }    
public void playWavFile(String fileName) {    
         playedWav = true;    
      }    
public boolean wavWasPlayed() {    
return playedWav;    
      }    
public void resetWav() {    
         playedWav = false;    
      }    
}

下面是mock对象:

public class MockSystemEnvironment implements Environmental {    
private long currentTime;    
public long getTime() {    
return currentTime;    
      }    
public void setTime(long currentTime) {    
this.currentTime = currentTime;    
      }    
public void playWavFile(String fileName) {    
         playedWav = true;    
      }    
public boolean wavWasPlayed() {    
return playedWav;    
      }    
public void resetWav() {    
         playedWav = false;    
      }    
}

1 2 下一页>>


[ 发表评论 ] 字体[  ] [ 打印 ] [ 进入博客 ] [ 进入论坛 ]  [ 推荐给朋友 ]
  相关文章
· 从数据库中读出图片并且显示的示例代码 (08-09) · 图解exe4j一步步生成exe执行文件的方法 (08-09)
· 精典文萃:Java编程中异常处理的优劣观 (08-09) · 用Java来监视系统进程的解决方案 (08-08)
· 使用Java程序实现随机验证码功能的实例 (08-08) · 使用Java来实现编辑器的Undo Redo功能 (08-07)
· Java新手学堂:Acegi框架技术详细介绍 (08-07) · 简单介绍 AspectJ 类的名称模式 (08-06)
· Java使用技巧:访问在接口中定义的常量 (08-06) · Java同步机制:sychronized对代码影响 (08-03)
  客户需求反馈表
* 姓  名:
更多资料  了解方案  认识厂商
* 单位名称:
* 联系电话:
* 电子邮件:
资讯 通信 IT产品 IT技术 信息化
专题:扭亏为盈 联想09年Q2翻番
·专题:Tech·Ed 2009微..
·直播:2009互联网大会..
·迅雷搜狐"互搏" 谁动了..
专题:诺基亚危局已现 或重蹈摩托覆辙
·六股势力角逐4G标准 中..
·专题:联通iPhone并不贵..
·专题:排排坐开商店 三..
商务演示需求分析 多媒体会议室必备投影
·VMware在京隆重举行2009..
·专题:笔记本频道10月热..
·专题:学生机市场 惠普 ..
专题:Tech.Ed 2009微软技术大会
·专题:2009 SYBASE 亚太..
·专题:微软新一代桌面操..
·专题:2009年第3届CSDN..