JUnit 是 Java 語(yǔ)言事實(shí)上的 標(biāo)準(zhǔn)單元測(cè)試庫(kù)。JUnit 4 是該庫(kù)三年以來(lái)具里程碑意義的一次發(fā)布。它的新特性主要是通過(guò)采用 Java 5 中的標(biāo)記(annotation)而不是利用子類、反射或命名機(jī)制來(lái)識(shí)別測(cè)試,從而簡(jiǎn)化測(cè)試。
在本文中,執(zhí)著的代碼測(cè)試人員 Elliotte Harold 以 JUnit 4 為例,詳細(xì)介紹了如何在自己的工作中使用這個(gè)新框架。注意,本文假設(shè)讀者具有 JUnit 的使用經(jīng)驗(yàn)。
JUnit 由 Kent Beck 和 Erich Gamma 開(kāi)發(fā),幾乎毫無(wú)疑問(wèn)是迄今所開(kāi)發(fā)的重要的第三方 Java 庫(kù)。正如 Martin Fowler 所說(shuō),“在軟件開(kāi)發(fā)領(lǐng)域,從來(lái)沒(méi)有如此少的代碼起到了如此重要的作用”。JUnit 引導(dǎo)并促進(jìn)了測(cè)試的盛行。由于 JUnit,Java 代碼變得更健壯,更可靠,bug 也比以前更少。JUnit(它本身的靈感來(lái)自 Smalltalk 的 SUnit)衍生了許多 xUnit 工具,將單元測(cè)試的優(yōu)勢(shì)應(yīng)用于各種語(yǔ)言。nUnit (.NET)、pyUnit (Python)、CppUnit (C++)、dUnit (Delphi) 以及其他工具,影響了各種平臺(tái)和語(yǔ)言上的程序員的測(cè)試工作。
然而,JUnit 僅僅是一個(gè)工具而已。真正的優(yōu)勢(shì)來(lái)自于 JUnit 所采用的思想和技術(shù),而不是框架本身。單元測(cè)試、測(cè)試先行的編程和測(cè)試驅(qū)動(dòng)的開(kāi)發(fā)并非都要在 JUnit 中實(shí)現(xiàn),任何比較 GUI 的編程都必須用 Swing 來(lái)完成。JUnit 本身的后一次更新差不多是三年以前了。盡管它被證明比大多數(shù)框架更健壯、更持久,但是也發(fā)現(xiàn)了 bug;而更重要的是,Java 不斷在發(fā)展。Java 語(yǔ)言現(xiàn)在支持泛型、枚舉、可變長(zhǎng)度參數(shù)列表和注釋,這些特性為可重用的框架設(shè)計(jì)帶來(lái)了新的可能。
JUnit 的停滯不前并沒(méi)有被那些想要廢棄它的程序員所打敗。挑戰(zhàn)者包括 Bill Venners 的 Artima SuiteRunner 以及 Cedric Beust 的 TestNG 等。這些庫(kù)有一些可圈可點(diǎn)的特性,但是都沒(méi)有達(dá)到 JUnit 的知名度和市場(chǎng)占有份額。它們都沒(méi)有在諸如 Ant、Maven 或 Eclipse 之類的產(chǎn)品中具有廣泛的開(kāi)箱即用支持。所以 Beck 和 Gamma 著手開(kāi)發(fā)了一個(gè)新版本的 JUnit,它利用 Java 5 的新特性(尤其是注釋)的優(yōu)勢(shì),使得單元測(cè)試比起用初的 JUnit 來(lái)說(shuō)更加簡(jiǎn)單。用 Beck 的話來(lái)說(shuō),“JUnit 4 的主題是通過(guò)進(jìn)一步簡(jiǎn)化 JUnit,鼓勵(lì)更多的開(kāi)發(fā)人員編寫更多的測(cè)試。”JUnit 4 盡管保持了與現(xiàn)有 JUnit 3.8 測(cè)試套件的向后兼容,但是它仍然承諾是自 JUnit 1.0 以來(lái) Java 單元測(cè)試方面重大的改進(jìn)。
注意:該框架的改進(jìn)是相當(dāng)前沿的。盡管 JUnit 4 的大輪廓很清晰,但是其細(xì)節(jié)仍然可以改變。這意味著本文是對(duì) JUnit 4 搶先看,而不是它的終效果。
測(cè)試方法
以前所有版本的 JUnit 都使用命名約定和反射來(lái)定位測(cè)試。例如,下面的代碼測(cè)試 1+1 等于 2:
clearcase/" target="_blank" >cccccc width="90%" align=center bgColor=#e7e9e9 border=1>
import junit.framework.TestCase;
public class AdditionTest extends TestCase {
private int x = 1;
private int y = 1;
public void testAddition() {
int z = x + y; assertEquals(2, z);
}
}
而在 JUnit 4 中,測(cè)試是由 @Test 注釋來(lái)識(shí)別的,如下所示:
import org.junit.Test;
import junit.framework.TestCase;
public class AdditionTest extends TestCase {
private int x = 1;
private int y = 1;
@Test public void testAddition() {
int z = x + y;
assertEquals(2, z);
}
}
使用注釋的優(yōu)點(diǎn)是不再需要將所有的方法命名為 testFoo()、testBar(),等等。例如,下面的方法也可以工作:
import org.junit.Test;
import junit.framework.TestCase;
public class AdditionTest extends TestCase {
private int x = 1;
private int y = 1;
@Test public void additionTest() {
int z = x + y; assertEquals(2, z);
}
}
下面這個(gè)方法也同樣能夠工作:
import org.junit.Test;
import junit.framework.TestCase;
public class AdditionTest extends TestCase {
private int x = 1;
private int y = 1;
@Test public void addition() {
int z = x + y; assertEquals(2, z);
}
}
這允許您遵循適合您的應(yīng)用程序的命名約定。例如,我介紹的一些例子采用的約定是,測(cè)試類對(duì)其測(cè)試方法使用與被測(cè)試的類相同的名稱。例如,List.contains() 由 ListTest.contains() 測(cè)試,List.add() 由 ListTest.addAll() 測(cè)試,等等。
TestCase 類仍然可以工作,但是您不再需要擴(kuò)展它了。只要您用 @Test 來(lái)注釋測(cè)試方法,可以將測(cè)試方法放到任何類中。但是您需要導(dǎo)入 junit.Assert 類以訪問(wèn)各種 assert 方法,如下所示:
import org.junit.Assert;
public class AdditionTest {
private int x = 1;
private int y = 1;
@Test public void addition() {
int z = x + y;
Assert.assertEquals(2, z);
}
}
您也可以使用 JDK 5 中新特性(static import),使得與以前版本一樣簡(jiǎn)單:
import static org.junit.Assert.assertEquals;
public class AdditionTest {
private int x = 1;
private int y = 1;
@Test public void addition() {
int z = x + y; assertEquals(2, z);
}
}
這種方法使得測(cè)試受保護(hù)的方法非常容易,因?yàn)闇y(cè)試案例類現(xiàn)在可以擴(kuò)展包含受保護(hù)方法的類了。