JAVA使用 MOCK 对象进行单元测试的实例讲解_mock某个对象的方法-程序员宅基地

技术标签: java  测试  单元测试  MOCK  

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


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

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");    
         }    
      }    
}

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

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());    
      }    
}

这就是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了。

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();    
     }    
}

编译并将其当做一个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测试的开发工具。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/c289054531/article/details/9187055

智能推荐

【浏览器】五大最好用的浏览器 最受欢迎的浏览器软件_火狐浏览器与edge-程序员宅基地

文章浏览阅读1.2k次,点赞30次,收藏27次。作为浏览器,它的界面简洁清爽,并且不管打开速度,还是加载速度都很快。除此之外,还有网页警示功能,会对打开的网页进行安全分析,若是网页存在安全问题,会提示用户。不管是哪款浏览器,只要适合自己使用,速度快稳定性强,安全性高,无广告无弹窗,功能齐全,那么就是比较好的浏览器。我们可以根据自己的情况,选择自己喜欢的浏览器,毕竟萝卜青菜各有所爱嘛。不同的浏览器提供不同的功能和性能,以满足用户对互联网浏览的不同需求。但是这里建议确定喜欢使用的浏览器后,我们可以将不需要的浏览器卸载,这样可以节省硬盘空间。_火狐浏览器与edge

VxWorks6.6安装-程序员宅基地

文章浏览阅读752次。在VMware10下安装VxWorks6.6虚拟机教程(上篇——准备环节)https://blog.csdn.net/kulala082/article/details/53887413在VMware10下安装VxWorks6.6虚拟机教程(中篇——FTP引导的bootrom.bin制作过程)https://blog.csdn.net/kulala082/article/details/53887920在VMware10下安装VxWorks6.6虚拟机教程(下篇——安装VxWorks 6.6_vxworks6.6

NR PDCP协议-程序员宅基地

文章浏览阅读1.6k次,点赞26次,收藏25次。PDCP:Packet Data Convergence Protocol主要功能:数据传输 PDCP SN的维护重排序和按序递交重复包丢弃头压缩加密解密完整性保护(相对于LTE,用户面数据属于可选的)基于discardtimer 的SDU丢弃PDCP重建_nr pdcp

从用户体验的角度说明产品的五个层次_选择一款产品,分析它的5个产品层次-程序员宅基地

文章浏览阅读3.9k次。战略层:解决有没有好产品的问题,要明确产品的定位和方向。能否直接或间接的满足用户的需求,是一个产品是否投入研发的关键。范围层:解决产品能不能用的问题,从功能和内容两个方,面来要求产品,兼顾功能性和实用性,使产品符合战略层目标,满足某种期望。结构层:解决产品好不好用的问题,主要从交互的角度对产品提出要求,简化使用流程。表现层:解决产品是否美观的问题,从视觉角度对产品内容进行整合,吸引用户眼球,提升用户_选择一款产品,分析它的5个产品层次

二维数组,字符串,字符数组_二维字符串数组-程序员宅基地

文章浏览阅读2.7k次。二维数组一.二维数组初始化【此处介绍4种方法】1.分行给二维数组赋初值,如int a[3][4]={{1,2,3,4},{5,6,7,8},{9,10,11,12}};2.可以将所有数据写在一个花括号内 如int a[3][4]={1,2,3,4,5,6,7,8,9,10,11,12};3.对部分元素赋初值int a[3][4]={{1},{5},{9}};对各行中的某一元素赋初值 in..._二维字符串数组

冠军斩获10万奖金!首届“域见杯”医检AI开发者大赛精彩落幕-程序员宅基地

文章浏览阅读1k次。首届“域见杯”医检AI开发者大赛精彩落幕。

随便推点

c语言发送网络请求,如何使用C+发出HTTP请求?-程序员宅基地

文章浏览阅读783次。Windows代码:#include#include#include#include#include#include#includeusingnamespacestd;#pragmacomment(lib,"ws2_32.lib")intmain(void){WSADATAwsaData;SOCKETSocket;SOCKADDR_INSockAddr;intli..._c网络请求

ccc计算机比赛如何报名,整理:加拿大的CCC是什么,怎么报名?-程序员宅基地

文章浏览阅读1.6k次。原标题:整理:加拿大的CCC是什么,怎么报名?CCC计算机竞赛(全称 Canadian Computing Competition)由加拿大滑铁卢大学与清华大学联合举办面向高中生的CS计算机竞赛,是加拿大历史悠久,影响深远的计算机竞赛。自1996年首界竞赛至今已经有24年历史,累积超过近10万名加拿大境内外高中生参赛。不仅如此,CCC证书也是作为留学背景提升、申请北美知名院校,极具含金量的计算机证..._ccc计算机 考试中可以使用

RK3568 学习笔记 : ubuntu 20.04 下 Linux-SDK 镜像烧写_rk3568刷linux-程序员宅基地

文章浏览阅读952次,点赞8次,收藏7次。RK3568 学习笔记 : ubuntu 20.04 下 Linux-SDK 镜像烧写_rk3568刷linux

Gradle是什么_gradle是干嘛的-程序员宅基地

文章浏览阅读1w次。以下总结作为本人学习笔记使用:Gradle是一个开源的自动构建工具,可以作为android studio工程的依赖管理和打包工具,包括三方库和jar包的依赖和链接, java和res等资源的编译和打包1、在gradle之前,android的构建工具是ant,之前eclipse用的就是ant2、gradle脚本不使用传统的xml文件,而是一种基于Groovy的动态DSL,而Groovy语言是一种基于jvm的动态语言,可以像写脚本一样去写项目构建规则3、Android Gradle Plugin_gradle是干嘛的

adb命令集锦-程序员宅基地

文章浏览阅读75次。adb 是什么?adb工具即Android Debug Bridge(安卓调试桥) tools。它就是一个命令行窗口,用于电脑端与模拟器或者真实设备交互。常用操作:把文件发送到android设备,或从android拖拽文件到本地在android设备上安装/卸载应用把APP的日志导出到本地截屏思考:上面说的这几点能否有其他方式替代,有什么区别?adb怎么使用?下载adb压缩包..._adb命令精简微信

【Java基础学习打卡15】分隔符、标识符与关键字_java分隔符有哪三种-程序员宅基地

文章浏览阅读1k次,点赞2次,收藏4次。本文将依托入门程序介绍 Java 代码中的分隔符、标识符和关键字,是基础中的基础。为之后的基础学习打基础。_java分隔符有哪三种

推荐文章

热门文章

相关标签