前言:本文章需要JUnit单元测试框架的基础知识,若读者还不具备,请阅读笔者的JUnit文章:http://ray-yui.iteye.com/blog/1914106
UnitTest系列文章:
使用JUnit开发单元测试:http://ray-yui.iteye.com/blog/1914106
使用DBUnit扩展JUnit:http://ray-yui.iteye.com/blog/1914979
使用EasyMock扩展Junithttp://ray-yui.iteye.com/blog/1916170
使用Spring TestContext测试Spring应用http://ray-yui.iteye.com/blog/1921424
使用Cobertura生成测试覆盖率报告http://ray-yui.iteye.com/blog/1921958
什么是Cactus?
Cactus是Apache下的一个开源测试Web层的框架,可以完成模拟J2EE的容器来进行测试,可以测试Servlet,JSP,Filter,EJB等等,以下图片为Cactus官方网站的原理图
为什么要使用Cactus?
当测试DAO时我们可以使用DBUnit来进行对数据库的隔离,当我们测试Service的时候可以使用EasyMock来模拟DAO的实现进行Service和DAO的隔离,虽然我们测试Servlet的时候也可以使用EasyMock来对Session,Request等来进行模拟,但实际上我们会发现,使用Mock来模拟Servlet只能测试简单的Servlet,例如Session,Request,Parameter等,对一些Servlet的返回值,返回了什么View等,测试起来还是力不从心,使用Cactus可以产生模拟的容器来对Servlet进行测试,大大减低了测试难度和提高了测试效率
Cactus使用:
增加Maven依赖
<dependency> <groupId>httpunit</groupId> <artifactId>httpunit</artifactId> <version>1.7</version> </dependency> <dependency> <groupId>org.apache.cactus</groupId> <artifactId>cactus.core.framework.uberjar.javaEE.15</artifactId> <version>1.8.1</version> </dependency>
在maven管理下的src/test/resources/中增加cactus.properties
cactus.contextURL=http://localhost:8080/test
以下为被测试的Servlet
package com.accentrix.ray.test.unit; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import com.accentrix.ray.entity.User; public class LoginServlet extends HttpServlet { /** * */ private static final long serialVersionUID = 6916692964267082204L; @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { String userName = req.getParameter("username"); if (userName == null || "".equals(userName.trim())) resp.sendRedirect("/error.jsp"); else req.getRequestDispatcher("/success.jsp").forward(req, resp); } }
以下为测试类
package com.accentrix.ray.test.unit; import java.io.IOException; import javax.servlet.ServletException; import org.apache.cactus.ServletTestCase; import org.apache.cactus.WebRequest; import org.xml.sax.SAXException; import com.meterware.httpunit.WebResponse; public class TestLoginServlet extends ServletTestCase { private LoginServlet loginServlet; /* * 由begin开始的方法,都被视为客户端方法,这里的操作 都应该是从 * 客户端发出的,例如客户端传递给服务端的参数就应该从begin方法 * 中传入,参数WebRequest为生成request而需要 */ public void beginLogin(WebRequest request) { // 通过request.addParameter()来添加parameter request.addParameter("username", "Ray"); } /* * 初始化对象 */ public void setUp() { loginServlet = new LoginServlet(); } /* * 测试doGet方法 */ public void testDoGet() throws ServletException, IOException { // 注意,此时request和response为ServletTestCase的属性 loginServlet.doGet(super.request, super.response); } /* * 由end开始的方法,都被视为服务器响应方法,这里操作 * 的应该是验证服务器传回来的结果,例如返回的JSP是否正确 * ,参数WebResponse为验证提供了大量的方法,例如getTable */ public void endDoGet(WebResponse response) throws SAXException { //返回已编译后的html页面的title assertEquals("test", response.getTitle()); //返回整个html的document assertEquals("test", response.getDOM().getElementById("Ray").getNodeValue()); //返回指定名称的元素 assertEquals(3, response.getElementsByTagName("input").length); } }
编写完毕后,还需要把项目布置到服务器当中,开启服务器后运行test case
注意事项:
1.在cactus.properties文件中,contextURL的值需要注意后面不能添加 '/',
例如http://localhost:8080/test/ <--这样是错误的,因为Cactus会寻找
http://localhost:8080/test/ServletRedirector来解释测试类
2.Cactus1.8.1版本目前还不支持Junit4的Annotation,所以开发这需要遵循
Junit3.8的规范来开发,例如@Before需要用setUp()来替换
3.编写test case时,beginXXX testXXX endXXX为一个组合,xxx为自定义名字,
不同组合的begin和end不会互相影响
4.在上述例子中endDoGet(WebResponse response)中WebResponse中的类型
为httpunit中的实现类com.meterware.httpunit.WebResponse,若开发者
使用的是Cactus中的org.apache.cactus.WebRequest时,将不能使用getTable(),
getTitle等方法
Cactus是功能非常强大的Web容器测试框架,除了测试Servlet还可以测试JSP,Filter,EJB等,当有实际需求时,可以上Cactus官方网站进行学习,上面都有非常详细的Demo和解释http://jakarta.apache.org/cactus/
总结:
笔者在学习Cactus时发现可以使用Jetty嵌入式服务器达到在测试前启动服务器的效果,但学习过后Demo并不能成功运行,所以请有兴趣的读者参考其他资料,笔者觉得在实际开发当中没有使用Cactus的必要,因为Web层是离前台最近的地方,通过界面点击进行参数等的观察都非常方便,测试的价值并没有Service和DAO的大,而且Cactus虽然功能强大,但易用性并不是太好.建议读者遇到实际需求时认真选择是否使用Cactus