我們前面提到過測(cè)試是一個(gè)不會(huì)中斷的過程。一旦你有了一個(gè)測(cè)試,你要一直確保其正常工作,以檢驗(yàn)?zāi)闼尤氲男碌墓ぷ鞔a。不要每隔幾天或后才運(yùn)行測(cè)試,每天你都應(yīng)該運(yùn)行一下測(cè)試代碼。這種投資很小,但可以確保你得到可以信賴的工作代碼。你的返工率降低了,你會(huì)有更多的時(shí)間編寫工作代碼。
不要認(rèn)為壓力大,不寫測(cè)試代碼。相反編寫測(cè)試代碼會(huì)使你的壓力逐漸減輕,應(yīng)為通過編寫測(cè)試代碼,你對(duì)類的行為有了確切的認(rèn)識(shí)。你會(huì)更快地編寫出有效率地工作代碼。下面是一些具體的編寫測(cè)試代碼的技巧或較好的實(shí)踐方法:
1. 不要用TestCase的構(gòu)造函數(shù)初始化Fixture,而要用setUp()和tearDown()方法。
2. 不要依賴或假定測(cè)試運(yùn)行的順序,因?yàn)镴Unit利用Vector保存測(cè)試方法。所以不同的平臺(tái)會(huì)按不同的順序從Vector中取出測(cè)試方法。
3. 避免編寫有副作用的TestCase。例如:如果隨后的測(cè)試依賴于某些特定的交易數(shù)據(jù),不要提交交易數(shù)據(jù)。簡(jiǎn)單的會(huì)滾可以了。
4. 當(dāng)繼承一個(gè)測(cè)試類時(shí),記得調(diào)用父類的setUp()和tearDown()方法。
5. 將測(cè)試代碼和工作代碼放在一起,一邊同步編譯和更新。(使用Ant中有支持junit的.)
6. 測(cè)試類和測(cè)試方法應(yīng)該有一致的命名方案。如在工作類名前加上test從而形成測(cè)試類名。
7. 確保測(cè)試與時(shí)間無關(guān),不要依賴使用過期的數(shù)據(jù)進(jìn)行測(cè)試。導(dǎo)致在隨后的維護(hù)過程中很難重現(xiàn)測(cè)試。
8. 如果你編寫的軟件面向國(guó)際市場(chǎng),編寫測(cè)試時(shí)要考慮國(guó)際化的因素。不要僅用母語的Locale進(jìn)行測(cè)試。
9. 盡可能地利用JUnit提供地assert/fail方法以及異常處理的方法,可以使代碼更為簡(jiǎn)潔。
10.測(cè)試要盡可能地小,執(zhí)行速度快。
事實(shí)上,JUnit還可用于集成測(cè)試,但我并沒涉及到,原因有兩個(gè):一是因?yàn)闆]有單元測(cè)試,集成測(cè)試無從談起。我們接受測(cè)試地概念已經(jīng)很不容易了,如果再引入集成測(cè)試會(huì)更困難。二是我比較懶,希望將集成測(cè)試的任務(wù)交給測(cè)試人員去做。在JUnit的上有一些相關(guān)的文章,有空大家可以翻一翻。
7.JUnit與J2EE
如果大家仔細(xì)考慮一下的話,會(huì)發(fā)現(xiàn),JUnit有自己的局限性,比如對(duì)圖形界面的測(cè)試,對(duì)servlet/JSP以及EJB的測(cè)試我們都沒有舉相關(guān)的例子。實(shí)際上,JUnit對(duì)于GUI界面,servlet/JSP,JavaBean以及EJB都有辦法測(cè)試。關(guān)于GUI的測(cè)試比較復(fù)雜,適合用一整篇文章來介紹。這里不多說了。
前面我們所做的測(cè)試實(shí)際上有一個(gè)隱含的環(huán)境,JVM我們的類需要這個(gè)JVM來執(zhí)行。而在J2EE框架中,servlet/JSP,EJB都要求有自己的運(yùn)行環(huán)境:Web Container和EJB Container。所以,要想對(duì)servlet/JSP,EJB進(jìn)行測(cè)試需要將其部署在相應(yīng)的Container中才能進(jìn)行測(cè)試。由于EJB不涉及UI的問題(除非EJB操作XML數(shù)據(jù),此時(shí)的測(cè)試代碼比較難寫,有可能需要你比較兩棵DOM樹是否含有相同的內(nèi)容)只要部署上去之后可以運(yùn)行測(cè)試代碼了。此時(shí)setUp()方法顯得特別有用,你可以在setUp()方法中利用JNDI查找特定的EJB。而在testXXX()方法中調(diào)用并測(cè)試這些EJB的方法。
這里所指的JavaBean同樣沒有UI的問題,比如,我們用JavaBean來訪問數(shù)據(jù)庫,或用JavaBean來包裹EJB。如果這類JavaBean沒有用到Container的提供的服務(wù),則可直接進(jìn)行測(cè)試,同我們前面所說的一般的類的測(cè)試方法一樣。如果這類JavaBean用到了Container的提供的服務(wù),則需要將其部署在Container中才能進(jìn)行測(cè)試。方法與EJB類似。
對(duì)于servlet/JSP的測(cè)試則比較棘手,有人建議在測(cè)試代碼中構(gòu)造HttpRequest和HttpResponse,然后進(jìn)行比較,這要求開發(fā)人員對(duì)HTTP協(xié)議以及servlet/JSP的內(nèi)部實(shí)現(xiàn)有比較深的認(rèn)識(shí)。我認(rèn)為這招不太現(xiàn)實(shí)。也有人提出使用。由于我對(duì)和HttpUnit 了解不多,所以無法做出合適的建議。希望各位先知們能不吝賜教()。
正是由于JUnit的開放性和簡(jiǎn)單易行,才會(huì)引出這篇介紹文章。但技術(shù)總在不斷地更新,而且我對(duì)測(cè)試并沒有非常深入的理解;我可以將一個(gè)復(fù)雜的概念簡(jiǎn)化成一句非常容易理解的話。但我的本意只是希望能降低開發(fā)人員步入測(cè)試領(lǐng)域的門檻,而不是要修改或重新定義一些概念。這一點(diǎn)是特別要強(qiáng)調(diào)的。后,如果有些兄弟姐妹能給我()指出一些注意事項(xiàng)或我對(duì)某些問題的理解有誤,我會(huì)非常感激的。