Junit和許多開源軟件項目集成在一起,但是Junit執(zhí)行多線程的單元測試有一些問題。這篇文章介紹Junit的一個擴(kuò)展類庫———GroboUtils,這個類庫被設(shè)計為來解決這些問題,并且使在Junit中進(jìn)行單元測試成為可能。對Junit和線程有一個基本的理解是有好處的,但對于本篇文章的讀者來說不是必需的。
介紹
如果你已經(jīng)在一個開源的Java項目上工作,或者讀了許多有關(guān)“極限編程”和其它“快速開發(fā)模式”的書籍,那么,你很有可能已經(jīng)聽說過有關(guān)Junit的事情。它是由Erich Gamma和Kent Beck編寫的,Junit是一個Java的自動測試的框架,它允許你為你的軟件定義的“單元測試”———不管是測試程序還是功能代碼,通常都是基于方法調(diào)用方法的。
Junit能在很多方面幫助你的開發(fā)團(tuán)隊———在一些文章中已經(jīng)包含了很多這方面的介紹。但從一個開者到另一個開發(fā)者,Junit實際上只專箸于兩件事:
1、它強(qiáng)制你使用自己的代碼。你的測試代碼只是作為你的產(chǎn)品代碼的客戶端,從客戶端的描述所獲得的對你的軟件的了解,能夠幫助你標(biāo)識出在API中的錯誤以及怎樣改進(jìn)代碼,使其終達(dá)到可以使用的目的。
2、它會給你對軟件中改變帶來信心,如果你的測試用例被中斷,你是立刻知道錯誤。在工作結(jié)束的時候,如果測試提示是綠色的,則代碼是正確,你可以自信的檢查它。
但是Junit不是解決所有軟件測試中問題,第三方的擴(kuò)展類庫,例如HttpUnit,JwebUnit,XMLUnit等,已經(jīng)認(rèn)識到這些框架中不足,并且通過添加功能彌補(bǔ)不足,這些不足之一是Junit不包含多線程的單元測試。
在這篇文章中,我們會看到一個很少有人知道的解決這個問題的擴(kuò)展類庫。我們通過建立Junit框架開始,并且運(yùn)行一個例子來展示Junit在線程沒試中的不足。在我們認(rèn)識了Junit在線程測試方面的不足之后,我們通過一個使用GroboUtils框架的例子來討論GroboUnitls
線程回顧
對于那些不熟悉線程的人來說,在這一點(diǎn)上是非常不安的(一點(diǎn)都不夸大),離開你的系統(tǒng),我們將對線做一個簡單的介紹。線程允許你的軟件有多個任務(wù),也是說可以同時可做兩件事情。
在Khalid Mugal和Rolf Rasmussen的書(A Programmer's Guide to Java Certification)中,對線程做了下面這樣的簡短描述:
一個線程是一個程序中的可執(zhí)行單元,它是被獨(dú)立執(zhí)行的。在運(yùn)行時,在程序中的線程有一個公共的內(nèi)存空間,因此,能夠共享數(shù)據(jù)和代碼;也是說,它們是輕量級的。它也共享正在運(yùn)行程序的進(jìn)程。
Java 線程使運(yùn)行時環(huán)境異步,它允許不同的任務(wù)同時被執(zhí)行。(p.272)
在web應(yīng)用程序中,許多用戶可能同時發(fā)請求給你的軟件。當(dāng)你寫單元測試對你的代碼進(jìn)行壓力測試時,你需要模擬許多并發(fā)事件,如果你在開發(fā)健壯的中間件,這樣做是尤其重要的。對于這些組件,使用線程測試是一個好的想法。
不幸的是,Junit在這方面是不足的。
有關(guān)Junit和多線程測試的問題
如果你想驗證下列代碼,你需要下載并安裝Junit。按著指示去做,以便能夠在Junit的網(wǎng)站能夠找到它。不要過分追求細(xì)節(jié),我們將簡要的介紹Junit是怎樣工作的。要寫一個Junit的測試,你必須首先創(chuàng)建一個擴(kuò)展于junit.framework.TestCase(Juint中的基本測試類)的測試類。
Main()方法和suite()方法被用啟動測試。無論是從命令行還是IDE集成開發(fā)環(huán)境窗口,必須確保junit.jar在你的CLASSPATH環(huán)境變量里指定。然后為BadExampleTest.Class類編譯運(yùn)行下列代碼:
import junit.framework.*;
public class BadExampleTest extends TestCase {
// For now, just verify that the test runs
public void testExampleThread()
throws Throwable {
System.out.println("Hello, World");
}
public static void main (String[] args) {
String[] name =
{ BadExampleTest.class.getName() };
junit.textui.TestRunner.main(name);
}
public static Test suite() {
return new TestSuite(
BadExampleTest.class);
}
}
運(yùn)行BadExampleTest來驗證所建立的每一件事情的正確性。一旦,main()被調(diào)用,Junit框架將自動的執(zhí)行任意一個用“test”開關(guān)命名的方法。繼續(xù)并試著運(yùn)行測試類。如果你正確的做了每一件事,它應(yīng)該在輸出窗口打印出“Hello World”。
現(xiàn)在,我們要給程序添加一個線程類。我將通過擴(kuò)展java.lang.Runnable接口來做這件事情。后,我們將改變策略,并且擴(kuò)展一個使線程自動創(chuàng)建的類。
在DelayedHello的構(gòu)造器中,我們創(chuàng)建一個新的線程并且調(diào)用它的start()方法。
import junit.framework.*;
public class BadExampleTest extends TestCase {
private Runnable runnable;
public class DelayedHello
implements Runnable {
private int count;
private Thread worker;
private DelayedHello(int count) {
this.count = count;
worker = new Thread(this);
worker.start();
}
public void run() {
try {
Thread.sleep(count);
System.out.println(
"Delayed Hello World");
} catch(InterruptedException e) {
e.printStackTrace();
}
}
}
public void testExampleThread()
throws Throwable {
System.out.println("Hello, World"); //1
runnable = new DelayedHello(5000); //2
System.out.println("Goodbye, World"); //3
}
public static void main (String[] args) {
String[] name =
{ BadExampleTest.class.getName() };
junit.textui.TestRunner.main(name);
}
public static Test suite() {
return new TestSuite(
BadExampleTest.class);
}
}