`
hereson2
  • 浏览: 451041 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

使用 MOCK 对象进行单元测试

阅读更多

1.出了什么问题?

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

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

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

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

2.现在需要Mock对象吗?

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

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

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

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

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

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

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

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

3.如何实现Mock对象?

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

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

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

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

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

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

真实的实现代码:

<ccid_nobr>
<ccid_code>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;
}
}</ccid_code>
</ccid_nobr>

下面是mock对象:

<ccid_code>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;
}
}</ccid_code>

下面是一个调用getTime的具体类:

<ccid_nobr>
<ccid_code>import java.util.Calendar;    

public class Checker {
private Environmental env;
public Checker(Environmental env) {
this.env = env;
}
public void reminder() {
Calendar cal = Calendar.getInstance();
cal.setTimeInMills(env.getTime());
int hour = cal.get(Calendar.HOUR_OF_DAY);
if(hour >= 17) {
env.playWavFile("quit_whistle.wav");
}
}
}</ccid_code>
</ccid_nobr>

使用env.getTime()的被测代码并不知道测试环境和真实环境之间的区别,因为它们都实现了相同的接口。现在,你可以借助mock对象,通过把时间设置为已知值,并检查行为是否如预期那样来编写测试。

<ccid_nobr>
<ccid_code>import java.util.Calendar;    
import junit.framework.TestCase;

public class TestChecker extends TestCase {
public void testQuittingTime() {
MockSystemEnvironment env = new MockSystemEnvironment();
Calendar cal = Calendar.getInstance();
cal.set(Calendar.YEAR, 2006);
cal.set(Calendar.MONTH, 11);
cal.set(Calendar.DAY_OF_MONTH,7);
cal.set(Calendar.HOUR_OF_DAY, 16);

cal.set(Calendar.MINUTE, 55);
long t1 = cal.getTimeInMillis();
env.setTime(t1);
Checker checker = new Checker(env);
checker.reminder();
assertFalse(env.wavWasPlayed());
t1 += (5*60*1000);
env.setTime(t1);
checker.reminder();
assertTrue(env.wavWasPlayed());
env.resetWav();
t1 += 2*60*60*1000;
env.setTime(t1);
checker.reminder();
assertTrue(env.wavWasPlayed());
}
}</ccid_code>
</ccid_nobr>

这就是mock对象的全部:伪装出真实世界的某些行为,使你可以集中精力测试好自己的代码。

4.好像有一些麻烦

如果每次都像上面那样自己写具体的mock对象,问题虽然解决了,但是好像有一些麻 烦,不要着急,已经有一些第三方现成的mock对象供我们使用了。使用Mock Object进行测试,主要是用来模拟那些在应用中不容易构造(如HttpServletRequest必须在Servlet容器中才能构造出来)或者比 较复杂的对象(如JDBC中的ResultSet对象)从而使测试顺利进行的工具。目前,在Java阵营中主要的Mock测试工具有 JMock,MockCreator,Mockrunner,EasyMock,MockMaker等,在微软的.Net阵营中主要是 Nmock,.NetMock等。

下面就以利用EasyMock模拟测试Servlet组件为例,代码如下: 编译并将其当做一个Test Case运行,会发现两个测试方法均测试成功。我们可以看到easymock已经帮助我们实现了一些servlet组件的mock对象,这样我们就可以摆 脱web容器和servlet容器来轻松的测试servlet了。

<ccid_nobr>
<ccid_code>import org.easymock.*;    
import junit.framework.*;
import javax.servlet.http.*;

public class MockRequestTest extends TestCase{
private MockControl control;
private HttpServletRequest mockRequest;
public void testMockRequest(){
//创建一个Mock HttpServletRequest的MockControl对象
control = MockControl.createControl(HttpServletRequest.class);
//获取一个Mock HttpServletRequest对象
mockRequest = (HttpServletRequest) control.getMock();
//设置期望调用的Mock HttpServletRequest对象的方法
mockRequest.getParameter("name");
//设置调用方法期望的返回值,并指定调用次数
//以下后两个参数表示最少调用一次,最多调用一次
control.setReturnValue("kongxx" ,1 ,1);
//设置Mock HttpServletRequest的状态,
//表示此Mock HttpServletRequest对象可以被使用
control.replay();
//使用断言检查调用
assertEquals("kongxx",mockRequest.getParameter("name"));
//验证期望的调用
control.verify();
}
}</ccid_code>
</ccid_nobr>

编译并将其当做一个Test Case运行,会发现两个测试方法均测试成功。我们可以看到easymock已经帮助我们实现了一些servlet组件的mock对象,这样我们就可以摆脱web容器和servlet容器来轻松的测试servlet了。

5.底层技术是什么?

让我们来回忆一下,如果用户使用C++和java的程序的生成,C++在最后的阶段 还需要连接才能生成一个整体程序,这在灵活性与java源代码的机制是不能比的,java的各个类是独立的,打包的那些类也是独立的,只有在加载进去才进 行连接,这在代码被加载进去的时候,我们还可以执行很多的动作,如插入一些相关的业务需求,这也是AOP的一个焦点,javassit代码库的实现类似于 这,正是利用这些,所以用java实现Mock对象是很简单的。

6.一些相关的资源

MockObject的主页 http://www.mockobjects.com/ 介绍了关键Mock Object的基本概念和目前在各个环境下主要的Mock测试工具。

JMock的主页http://www.jmock.org/ 可以获取JMock的最新代码和开发包,以及一些说明文档。

EasyMock的主页http://www.easymock.org/ 可以获取JMock的最新代码和开发包,以及一些说明文档。

NMock的主页http://www.nmock.org/ 介绍了在Microsoft .Net平台上进行Mock测试的开发工具。


转自:http://tech.ccidnet.com/art/3539/20070809/1172211_1.html
分享到:
评论

相关推荐

    使用Mock对象进行单元测试(Rhino Mocks)

    有关如何使用模拟对象编写单元测试的基本介绍(Rhino Mocks)。

    有效使用Mock编写java单元测试

    Java单元测试对于开发人员质量保证至关重要,尤其当面对一团乱码的遗留代码时,没有高覆盖率的单元测试做保障,没人敢轻易对代码进行重构。然而单元测试的编写也不是一件容易的事情,除非使用TDD方式,否则编写出...

    mock-all-things:使用 mock 进行 python 单元测试的示例

    模拟一切使用 mock 进行 python 单元测试的示例模拟 python 2.7: : (这里也有很多很好的例子) 从图书馆的主页: “mock 是一个用于在 Python 中进行测试的库。它允许您用模拟对象替换被测系统的某些部分,并对...

    使用PythonMock类进行单元测试

    数据类型、模型或节点——这些都只是mock对象可承担的角色。但mock在单元测试中扮演一个什么角色呢?有时,你需要为单元测试的初始设置准备一些“其他”的代码资源。但这些资源兴许会不可用,不稳定,或者是使用起来...

    基于模拟对象MockObject的单元测试研究.pdf

    不错的文档,提供给科研人员

    使用模仿对象进行单元测试

    北京火龙果软件工程技术中心模仿对象(Mockobject)是为起中介者作用的对象编写单元测试的有用方法。测试对象调用模仿域对象(它只断言以正确的次序用期望的参数调用了正确的方法),而不是调用实际域对象。然而,当...

    easymock的使用,含demo

    EasyMock 是一套通过简单的方法对于指定的接口或类生成 Mock 对象的类库,它能利用对接口或类的模拟来辅助单元测试。本文将对 EasyMock 的功能和原理进行介绍,并通过示例来说明如何使用 EasyMock 进行单元测试。 ...

    模拟测试辅助工具easyMock.zip

    EasyMock 是一套通过简单的方法对于指定的接口或类生成 Mock 对象的类库,它能利用对接口或类的模拟来辅助...本文将向您展示如何使用 EasyMock 进行单元测试,并对 EasyMock 的原理进行分析。 标签:easyMock

    使用配置文件定义Mock对象,创建高效、灵活的测试用例

    火龙果软件工程技术中心 本文内容包括:1.Mock对象的创建方法2.EasyMock使用示例3.利用XML文件配置Mock对象4....使单元测试顺利进行。然而,Mock方法在辅助测试的同时,也给开发或测试人员带来额

    【mock】打桩说明文档及Jar包

    在java web应用中,进行用例实现时,很多情况难以模拟,比如数据库用例,如果直接通过连接数据库进行测试,导致用例对环境依赖很大,这时,可以通过mock技术可以模拟构造数据环境,从而进行单元测试,这里提供有实现...

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

    在具体的使用场景中,例如,对于一些不容易构造或者获取的对象(如HttpServletRequest必须在Servlet容器中才能构造出来,或者JDBC中的ResultSet对象),Mockito可以创建一个虚拟的对象(即Mock对象)进行测试。...

    单元测试之Stub和Mock

    我们需要用到Stub和Mock来模拟这些外部依赖的对象,从而控制它们  实例  Analyze类会检查filename的长度,如果小于8,我们就会使用一个实现IWebService的类来记录错误.  我们需要给Analyze方法写单元测试

    转:google mock C++单元测试框架-奋飞的菜鸟-ChinaUnix博客1

    引入你要用到的GoogleMock名称.除宏或其它特别提到的之外所有GoogleMock名称都位于*testing*命名空间之下.建立模拟对象(Mock Obj

    Python mock的基本使用介绍.md

    Python mock的基本使用介绍。Mock即模拟的意思。在Python中,提供了基于单元测试的Mock模块,它的主要作用是使用mock对象替代掉指定的Python对象,以达到模拟对象功能的行为。

    单元测试junit-4.13.1.zip

    通常来说,程式设计师每修改一次程式就会进行最少一次单元测试,在编写程式的过程中前后很可能要进行多次单元测试,以证实程式达到 软件规格书 要求的工作目标,没有 程序错误 ;虽然单元测试不是必须的,但也不坏...

    使用JMockit编写java单元测试

    引用单元测试中mock的使用及mock神器jmockit实践中的java单元测试中各种Mock框架对比,就能明白JMockit有多么强大:JMockit是基于JavaSE5中的java.lang.instrument包开发,内部使用ASM库来动态修改java的字节码,...

    easymock-4.2.jar

    EasyMock 是一套通过简单的方法对于指定的接口或类生成 Mock 对象的类库,它能利用对接口或类的模拟来辅助单元测试。 Mock 方法是单元测试中常见的一种技术,它的主要作用是模拟一些在应用中不容易构造或者比较复杂...

    Pascal Mock-开源

    用于Delphi / Kylix / Free Pascal的模拟对象库。 通过该库,可以轻松创建和使用Mock对象进行单元测试。 独立于单元测试框架。

Global site tag (gtag.js) - Google Analytics