单元测试是XP 极力推荐的测试驱动开发模式,是保证软件质量的重要方法。尽管如此,对许多
类的单元测试仍然是极其困难的,例如,对数据库操作的类进行测试,如果不准备好数据库环境以
及相关测试数据,是很难进行单元测试的;再例如,对需要运行在容器内的Servlet 或EJB 组件,脱
离了容器也难于测试。
幸运的是,Mock Object 可以用来模拟一些我们需要的类,这些对象被称之为模仿对象,在单元
测试中它们特别有价值。
Mock Object 用于模仿真实对象的方法调用,从而使得测试不需要真正的依赖对象。Mock Object
只为某个特定的测试用例的场景提供刚好满足需要的最少功能。它们还可以模拟错误的条件,例如
抛出指定的异常等。
目前,有许多可用的Mock 类库可供我们选择。一些Mock 库提供了常见的模仿对象,例如:
HttpServletRequest,而另一些Mock 库则提供了动态生成模仿对象的功能,本文将讨论使用EasyMock
动态生成模仿对象以便应用于单元测试。
到目前为止,EasyMock 提供了1.2 版本和2.0 版本,2.0 版本仅支持Java SE 5.0,本例中,我们
选择EasyMock 1.2 for Java 1.3 版本进行测试,可以从http://www.easymock.org 下载合适的版本。
我们首先来看一个用户验证的LoginServlet 类:
- /**
- *LoginServlet.java
- */
- packagecom.javaeedev.test.mock;
- importjava.io.*;
- importjavax.servlet.*;
- importjavax.servlet.http.*;
-
- publicclassLoginServletextendsHttpServlet{
- protectedvoiddoPost(HttpServletRequestrequest,HttpServletResponseresponse)
- throwsServletException,IOException{
- Stringusername=request.getParameter("username");
- Stringpassword=request.getParameter("password");
- //checkusername&password:
- if("admin".equals(username)&&"123456".equals(password)){
- ServletContextcontext=getServletContext();
- RequestDispatcherdispatcher=context.getNamedDispatcher("dispatcher");
- dispatcher.forward(request,response);
- }else{
- thrownewRuntimeException("Loginfailed.");
- }
- }
- }
这个Servlet 实现简单的用户验证的功能,若用户名和口令匹配“admin”和“123456”,则请求
被转发到指定的dispatcher 上,否则,直接抛出RuntimeException。
为了测试doPost()方法,我们需要模拟HttpServletRequest,ServletContext 和RequestDispatcher
对象,以便脱离J2EE 容器来测试这个Servlet。
我们建立TestCase,名为LoginServletTest:
- publicclassLoginServletTestextendsTestCase{
- }
我们首先测试当用户名和口令验证失败的情形, 演示如何使用EasyMock 来模拟
HttpServletRequest 对象:
- publicvoidtestLoginFailed()throwsException{
- MockControlmc=MockControl.createControl(HttpServletRequest.class);
- HttpServletRequestrequest=(HttpServletRequest)mc.getMock();
- //setMockObjectbehavior:
- request.getParameter("username");
- mc.setReturnValue("admin",1);
- request.getParameter("password");
- mc.setReturnValue("1234",1);
- //ok,allbehaviorsareset!
- mc.replay();
- //nowstarttest:
- LoginServletservlet=newLoginServlet();
- try{
- servlet.doPost(request,null);
- fail("Notcaughtexception!");
- }catch(RuntimeExceptionre){
- assertEquals("Loginfailed.",re.getMessage());
- }
- //verify:
- mc.verify();
- }
仔细观察测试代码,使用EasyMock 来创建一个Mock 对象需要首先创建一个MockControl:
- MockControlmc=MockControl.createControl(HttpServletRequest.class);
然后,即可获得MockControl 创建的Mock 对象:
- HttpServletRequestrequest=(HttpServletRequest)mc.getMock();
下一步,我们需要“录制”Mock 对象的预期行为。在LoginServlet 中, 先后调用了
request.getParameter("username") 和request.getParameter("password") 两个方法, 因此, 需要在
MockControl 中设置这两次调用后的指定返回值。我们期望返回的值为“admin”和“1234”:
- request.getParameter("username");//期望下面的测试将调用此方法,参数为"username"
- mc.setReturnValue("admin",1);//期望返回值为"admin",仅调用1次
- request.getParameter("password");//期望下面的测试将调用此方法,参数为"password"
- mc.setReturnValue("1234",1);//期望返回值为"1234",仅调用1次
紧接着,调用mc.replay(),表示Mock 对象“录制”完毕,可以开始按照我们设定的方式运行,
我们对LoginServlet 进行测试,并预期会产生一个RuntimeException:
- LoginServletservlet=newLoginServlet();
- try{
- servlet.doPost(request,null);
- fail("Notcaughtexception!");
- }catch(RuntimeExceptionre){
- assertEquals("Loginfailed.",re.getMessage());
- }
由于本次测试的目的是检查当用户名和口令验证失败后, LoginServlet 是否会抛出
RuntimeException,因此,response 对象对测试没有影响,我们不需要模拟它,仅仅传入null 即可。
最后,调用mc.verify()检查Mock 对象是否按照预期的方法调用正常运行了。
运行JUnit,测试通过!表示我们的Mock 对象正确工作了!
下一步,我们来测试当用户名和口令匹配时,LoginServlet 应当把请求转发给指定的
RequestDispatcher。在这个测试用例中,我们除了需要HttpServletRequest Mock 对象外,还需要模拟
ServletContext 和RequestDispatcher 对象:
- MockControlrequestCtrl=MockControl.createControl(HttpServletRequest.class);
- HttpServletRequestrequestObj=(HttpServletRequest)requestCtrl.getMock();
- MockControlcontextCtrl=MockControl.createControl(ServletContext.class);
- finalServletContextcontextObj=(ServletContext)contextCtrl.getMock();
- MockControldispatcherCtrl=MockControl.createControl(RequestDispatcher.class);
- RequestDispatcherdispatcherObj=(RequestDispatcher)dispatcherCtrl.getMock();
按照doPost()的语句顺序,我们设定Mock 对象指定的行为:
- requestObj.getParameter("username");
- requestCtrl.setReturnValue("admin",1);
- requestObj.getParameter("password");
- requestCtrl.setReturnValue("123456",1);
- contextObj.getNamedDispatcher("dispatcher");
- contextCtrl.setReturnValue(dispatcherObj,1);
- dispatcherObj.forward(requestObj,null);
- dispatcherCtrl.setVoidCallable(1);
- requestCtrl.replay();
- contextCtrl.replay();
- dispatcherCtrl.replay();
然后,测试doPost()方法,这里,为了让getServletContext()方法返回我们创建的ServletContext
Mock 对象,我们定义一个匿名类并覆写getServletContext()方法:
- LoginServletservlet=newLoginServlet(){
- publicServletContextgetServletContext(){
- returncontextObj;
- }
- };
- servlet.doPost(requestObj,null);
最后,检查所有Mock 对象的状态:
- requestCtrl.verify();
- contextCtrl.verify();
- dispatcherCtrl.verify();
运行JUnit,测试通过!
倘若LoginServlet 的代码有误, 例如, 将context.getNamedDispatcher("dispatcher") 误写为
context.getNamedDispatcher("dispatcher2"),则测试失败,JUnit 报告:
- junit.framework.AssertionFailedError:
- UnexpectedmethodcallgetNamedDispatcher("dispatcher2"):
- getNamedDispatcher("dispatcher2"):expected:0,actual:1
- getNamedDispatcher("dispatcher"):expected:1,actual:0
- at...
完整的LoginServletTest 代码如下:
- /**
- *LoginServletTest.java
- */
- packagecom.javaeedev.test.mock;
-
- importjavax.servlet.*;
- importjavax.servlet.http.*;
- importorg.easymock.*;
- importjunit.framework.TestCase;
-
- publicclassLoginServletTestextendsTestCase{
- publicvoidtestLoginFailed()throwsException{
- MockControlmc=MockControl.createControl(HttpServletRequest.class);
- HttpServletRequestrequest=(HttpServletRequest)mc.getMock();
- //setMockObjectbehavior:
- request.getParameter("username");
- mc.setReturnValue("admin",1);
- request.getParameter("password");
- mc.setReturnValue("1234",1);
- //ok,allbehaviorsareset!
- mc.replay();
- //nowstarttest:
- LoginServletservlet=newLoginServlet();
- try{
- servlet.doPost(request,null);
- fail("Notcaughtexception!");
- }catch(RuntimeExceptionre){
- assertEquals("Loginfailed.",re.getMessage());
- }
- //verify:
- mc.verify();
- }
- publicvoidtestLoginOK()throwsException{
- //createmock:
- MockControlrequestCtrl=MockControl.createControl(HttpServletRequest.class);
- HttpServletRequestrequestObj=(HttpServletRequest)requestCtrl.getMock();
- MockControlcontextCtrl=MockControl.createControl(ServletContext.class);
- finalServletContextcontextObj=(ServletContext)contextCtrl.getMock();
- MockControldispatcherCtrl=
- MockControl.createControl(RequestDispatcher.class);
- RequestDispatcherdispatcherObj=(RequestDispatcher)dispatcherCtrl.getMock();
- //setbehavior:
- requestObj.getParameter("username");
- requestCtrl.setReturnValue("admin",1);
- requestObj.getParameter("password");
- requestCtrl.setReturnValue("123456",1);
- contextObj.getNamedDispatcher("dispatcher");
- contextCtrl.setReturnValue(dispatcherObj,1);
- dispatcherObj.forward(requestObj,null);
- dispatcherCtrl.setVoidCallable(1);
- //done!
- requestCtrl.replay();
- contextCtrl.replay();
- dispatcherCtrl.replay();
- //test:
- LoginServletservlet=newLoginServlet(){
- publicServletContextgetServletContext(){
- returncontextObj;
- }
- };
- servlet.doPost(requestObj,null);
- //verify:
- requestCtrl.verify();
- contextCtrl.verify();
- dispatcherCtrl.verify();
- }
- }
分享到:
相关推荐
windows环境部署easymock,有用到的全部文件和安装说明,包换node.js,redis,mongdb,easymock源码
个人作品 5-Mock+EasyMock.rar
手动的构造 Mock 对象会给开发人员带来额外的编码量,而且这些为创建 Mock 对象而编写的代码很有可能引入错误。目前,有许多开源项目对动态构建 Mock 对象提供了支持,这些项目能够根据现有的接口或类动态生成,这样...
几个例子展示如何使用EasyMock进行单元测试
NULL 博文链接:https://bruceliu02.iteye.com/blog/1775446
EasyMock 是一套通过简单的方法对于指定的接口或类生成 Mock 对象的类库,它能利用对接口或类的模拟来辅助单元测试。 Mock 方法是单元测试中常见的一种技术,它的主要作用是模拟一些在应用中不容易构造或者比较...
EasyMock 是一套通过简单的方法对于指定的接口或类生成 Mock 对象的类库,它能利用对接口或类的模拟来辅助单元测试。 Mock 方法是单元测试中常见的一种技术,它的主要作用是模拟一些在应用中不容易构造或者比较复杂...
EasyMock 是一套通过简单的方法对于指定的接口或类生成 Mock 对象的类库,它能利用对接口或类的模拟来辅助单元测试。本文将对 EasyMock 的功能和原理进行介绍,并通过示例来说明如何使用 EasyMock 进行单元测试。 ...
EasyMock + junit 实例
easymock2 EasyMock使用简明手册
EasyMock用到的objenesis
Easy Mock 是一个可视化,并且能快速生成 模拟数据 的持久化服务
easy mock for java
什么时候需要Mock对象 什么是EasyMock EasyMock的优点和缺点 运行EasyMock需要的资源 EasyMock的基本使用步骤 什么是PowerMock 运行PowerMock需要的资源 PowerMock的扩展功能
EasyMock单元测试的扩展; EasyMock简介(抽象类接口做测试); EasyMock来进行测试; EasyMock如何打桩; EasyMock实践指南;...EasyMock使用技巧;...EasyMock使用简明手册;...EasyMock使用说明;...用Mock object进行隔离测试;
它提供对接口的模拟,能够通过录制、回放、检查三步来完成大体的测试过程,可以验证方法的调用种类、次数、顺序,可以令 Mock 对象返回指定的值或抛出指定异常。通过 EasyMock,我们可以方便的构造 Mock 对象从而使...
NULL 博文链接:https://zhangzhenting.iteye.com/blog/1829416
EasyMock 是一种模拟测试的框架,用于辅助模拟测试。当在测试过程中一些复杂的对象生成相当麻烦、费时或者根本无法生成时,可以用模拟的...原因是 EasyMock 在实现时为每个 class mock 对象提供了内建的以上三个方法。
easymock2.4 软件测试 mock