對于Java組件開發(fā)者來說,他們都盼望擁有一組能夠?qū)M件開發(fā)提供全面測試功能的好用的單元測試。一直以來,與測試獨(dú)立的Java對象相比,測試傳統(tǒng)型J2EE Web組件是一項(xiàng)更為困難的任務(wù),因?yàn)閃eb組件必須運(yùn)行在某種服務(wù)器平臺(tái)上并且它們還要與基于HTTP的Web交互細(xì)節(jié)相聯(lián)系。
易測性(在框架中測試每個(gè)組件而不管其具體種類)是Spring框架所提倡的關(guān)鍵原則之一。從這一角度看,Spring是對核心J2EE模型的一個(gè)重大改進(jìn)—在以前情況下,在容器外進(jìn)行組件測試是很難實(shí)現(xiàn)的,而且即使是容器內(nèi)測試也往往要求復(fù)雜的安裝過程。
本文正是想集中探討Spring的易測性特征—它能使得對Web組件進(jìn)行單元測試象測試普通Java對象(POJO)一樣容易。
一、Spring Mock類簡介
Mock對象是一個(gè)術(shù)語,原來主要流行于eXtreme程序員和JUnit小組中。在單元測試上下文中,一個(gè)mock對象是指這樣的一個(gè)對象——它能夠用一些“虛構(gòu)的占位符”功能來“模擬”實(shí)現(xiàn)一些對象接口。在測試過程中,這些虛構(gòu)的占位符對象可用簡單方式來模仿對于一個(gè)組件的期望的行為和結(jié)果,從而讓你專注于組件本身的徹底測試而不用擔(dān)心其它依賴性問題。
Spring從J2EE的Web端為每個(gè)關(guān)鍵接口提供了一個(gè)mock實(shí)現(xiàn):
MockHttpServletRequest—幾乎每個(gè)單元測試中都要使用這個(gè)類,它是J2EE Web應(yīng)用程序常用的接口HttpServletRequest的mock實(shí)現(xiàn)。
MockHttpServletResponse—此對象用于HttpServletResponse接口的mock實(shí)現(xiàn)。
MockHttpSession—這是另外一個(gè)經(jīng)常使用的mock對象(后文將討論此類在會(huì)話綁定處理中的應(yīng)用)。
DelegatingServletInputStream—這個(gè)對象用于ServletInputStream接口的mock實(shí)現(xiàn)。
DelegatingServletOutputStream—這個(gè)對象將代理ServletOutputStream實(shí)現(xiàn)。在需要攔截和分析寫向一個(gè)輸出流的內(nèi)容時(shí),你可以使用它。
總之,在實(shí)現(xiàn)你自己的測試控制器時(shí),上面這些對象是為有用的。然而,Spring也提供了下列相應(yīng)于其它不太常用的組件的mock實(shí)現(xiàn)(如果你是一個(gè)底層API開發(fā)者,那么你可能會(huì)找到其各自的相應(yīng)用法):
MockExpressionEvaluator—這個(gè)mock對象主要應(yīng)用于你想開發(fā)并測試你自己的基于JSTL的標(biāo)簽庫時(shí)。
MockFilterConfig—這是FilterConfig接口的一個(gè)mock實(shí)現(xiàn)。
MockPageContext—這是JSP PageContext接口的一個(gè)mock實(shí)現(xiàn)。你會(huì)發(fā)現(xiàn)這個(gè)對象的使用有利于測試預(yù)編譯的JSP。
MockRequestDispatcher—RequestDispatcher接口的一個(gè)mock實(shí)現(xiàn),你主要在其它mock對象內(nèi)使用它。
MockServletConfig—這是ServletConfig接口的一個(gè)mock實(shí)現(xiàn)。在單元測試某種Web組件(例如Struts框架所提供的Web組件)時(shí),要求你設(shè)置由MockServletContext所實(shí)現(xiàn)的ServletConfig和ServletContext接口。
那么,我們該如何使用這些mock對象呢?我們知道,HttpServletRequest是一個(gè)持有描述HTTP參數(shù)的固定值的組件,而正是這些參數(shù)驅(qū)動(dòng)Web組件的功能。MockHttpServletRequest,作為HttpServletRequest接口的一個(gè)實(shí)現(xiàn),允許你設(shè)置這些不可改變的參數(shù)。在典型的Web組件測試情形下,你可以實(shí)例化這個(gè)對象并按如下方式設(shè)置其中的任何參數(shù):
//指定表單方法和表單行為
MockHttpServletRequest request = new MockHttpServletRequest("GET", "/main.app");
request.addParameter("choice", expanded);
request.addParameter("contextMenu", "left");
同樣地,你可以實(shí)例化并全面地控制和分析HttpResponse和HttpSession對象。接下來,讓我們簡要觀察Spring所提供的特定的JUnit框架擴(kuò)展。
二、JUnit框架擴(kuò)展
Spring提供了下列一些特定的JUnit框架擴(kuò)展:
AbstractDependencyInjectionSpringContextTests—這是一個(gè)針對所有測試的超類,其具體使用依賴于Spring上下文。
AbstractSpringContextTests—這是一個(gè)針對所有的JUnit測試情形的超類。它使用一個(gè)Spring上下文。并且,一般在測試中不是直接使用它,而是使用AbstractDependencyInjectionSpringContextTests或者AbstractTransactionalSpringContextTests這樣的派生類。
AbstractTransactionalSpringContextTests—這是一個(gè)針對所有測試的超類,我們一般把它應(yīng)用在事務(wù)相關(guān)的測試中。注意,一旦完成每個(gè)測試它會(huì)正常地回滾事務(wù);而且你需要重載onSetUpInTransaction和onTearDownInTransaction方法以便手工開始并提交事務(wù)。
AbstractTransactionalDataSourceSpringContextTests—這是AbstractTransactionalSpringContextTests的一個(gè)子類,它使用了Spring的基于JDBC的jdbcTemplate工具類。
所有上面這些擴(kuò)展將極大程度地簡化在測試時(shí)對于相關(guān)操作的依賴性注入和事務(wù)管理。