JBoss Seam框架下的單元測試
想寫這篇文章很久了,因為 Seam 下的單元測試一直存在一些配置方面的問題,而且 Seam 框架提倡最多的是整合測試。所以關于單元測試,無論是 Seam 的參考手冊還是其他參考書(Seam In Action, etc.)里都沒有詳細介紹和例子??偹苤?,單元測試是保證質量或 TDD 必不可少的部分,對項目的重要性不言而喻。本文以實例說明了 Seam 的單元測試如何進行,有不對之處請大家指出 :-)
一. Seam 下的單元測試的難點與問題
1. Seam 框架依賴于容器環(huán)境
不只是 Seam,現(xiàn)在的 Web 框架很少有可以在純 JavaSE 環(huán)境中可運行的,所以脫離了容器環(huán)境,框架是不能運行的,其實也是沒有意義的。只要涉及到了容器,就會帶來很多的問題,而且因為容器的不同,問題錯綜復雜。
2. 容器內 / 容器外測試的復雜度
這里說的容器內測試就是在部署應用后在容器的環(huán)境里拿到應用的上下文,也就是在應用運行時在同一環(huán)境(容器內)拿到待測實例并進行測試。換句話說,測試用例與實現(xiàn)都是運行在容器的環(huán)境里。而容器外測試則是測試環(huán)境通過遠程訪問容器,拿到容器內的 Remote Seam Components / Remote EJB Session Beans 實例,從而在容器外的測試環(huán)境下進行測試。
3. JBoss 社區(qū)產品的整體性
Seam 是 JBoss 的,jBPM 是 JBoss 的,等等 Seam 相關的依賴環(huán)境都是 JBoss 的。所以,容器不是 JBoss 的的話,兩個字:麻煩。雖然現(xiàn)在的 Seam(ver 2.1.1) 在文檔 / 示例中說明了如何在非 JBoss AS 中配置與使用 Seam,但據筆者目前的嘗試,要想順利使用 Seam 的 Advanced 功能,JBoss AS / HIbernate 不用都不行。這也要求了要做 Seam 的單元測試,就必須對很多 JBoss 產品有一定的了解。
二. 解決問題
歸根結底,Seam 下的單元測試只所以不好做,還是難在了配置上。要配置好一個‘類容器’的環(huán)境才能讓 Seam 跑起來。
1. 搭建‘類容器’環(huán)境
從配置上來說,我們首先要搭建一個容器環(huán)境。把下載下來的 Seam 解壓,把 bootstrap 里面找到 conf 與 deploy 這兩個目錄并 copy 到測試環(huán)境目錄下。在測試環(huán)境目錄下建立 WEB-INF 目錄,并把 Seam 的核心配置文件 components.xml 放置到這里,還有 JSF、Web 部署描述符等需要的都放置到 WEB-INF 目錄里。
2. 在測試用例中拿到待測 Seam 組件實例
從代碼上來說,拿到待測組件實例是關鍵。在測試代碼中,可以通過 org.jboss.seam.Component.getInstance() 方法從 Seam 框架中獲取需要的組件實例,這是單元測試能否成功的關鍵。
3. 初始化 Seam 框架
在測試代碼中需要手動初始化 Seam 框架與應用,測試結束后也同樣需要手動結束框架與應用,其都是通過 org.jboss.seam.context.Lifecycle 進行操作的。
4. 測試框架的選擇
這不是相對于 Seam 單元測試的關鍵問題,但是,恰但地選擇測試框架對于項目而言是很重要的。總所周知,目前最常用的兩個測試框架 JUnit 與 TestNG 各有千秋,這里我簡單總結一下:
JUnit:適合小型項目,對于簡單的單元測試可以很好的發(fā)揮其‘小而強大’的優(yōu)勢,主流 IDE 都有其功能或插件。但要進行整合測試(集成界面)就困難了。
TestNG:適合任何項目,IDE 支持不是很好,但是 JUnit 能做的它都能做,而且更細致,適用場景更廣泛。
三. 示例
筆者是用 NetBeans IDE 與 Maven 搭建的項目,Seam 下的單元測試的具體配置和示例代碼片斷如下:

初始化:
if (!Lifecycle.isApplicationInitialized()) {
Lifecycle.beginApplication(new HashMap
new Initialization(new MockServletContext()).create().init();
Lifecycle.beginCall();
}
獲取待測組件實例:
processDefTransformer = (ProcessDefTransformService) Component.
getInstance(ProcessDefTransformer.class);
assertNotNull(processDefTransformer);
結束:
Lifecycle.endApplication();
四. 總結
本文筆者總結了 Seam 下單元測試的難點與問題,也提出了一種解決方案。隨著 Seam 框架的日漸穩(wěn)定與成熟,以及 WebBeans 規(guī)范的發(fā)布,相信 Java Web,Java EE 將迎來一個嶄新的時代!
您正在閱讀的是:JBoss Seam框架下的單元測試【編輯推薦】