1、概述
Junit測(cè)試是程序員測(cè)試,即所謂白盒測(cè)試,因?yàn)槌绦騿T知道被測(cè)試的軟件如何(How)完成功能和完成什么樣(What)的功能。
Junit本質(zhì)上是一套框架,即開(kāi)發(fā)者制定了一套條條框框,遵循這此條條框框要求編寫(xiě)測(cè)試代碼,如繼承某個(gè)類(lèi),實(shí)現(xiàn)某個(gè)接口,可以用Junit進(jìn)行自動(dòng)測(cè)試了。
由于Junit相對(duì)獨(dú)立于所編寫(xiě)的代碼,可以測(cè)試代碼的編寫(xiě)可以先于實(shí)現(xiàn)代碼的編寫(xiě),XP 中推崇的 test first design的實(shí)現(xiàn)有了現(xiàn)成的手段:用Junit寫(xiě)測(cè)試代碼,寫(xiě)實(shí)現(xiàn)代碼,運(yùn)行測(cè)試,測(cè)試失敗,修改實(shí)現(xiàn)代碼,再運(yùn)行測(cè)試,直到測(cè)試成功。以后對(duì)代碼的修改和優(yōu)化,運(yùn)行測(cè)試成功,則修改成功。
Java 下的 team 開(kāi)發(fā),采用 cvs(版本控制) + ant(項(xiàng)目管理) + junit(集成測(cè)試) 的模式時(shí),通過(guò)對(duì)ant的配置,可以很簡(jiǎn)單地實(shí)現(xiàn)測(cè)試自動(dòng)化。
對(duì)不同性質(zhì)的被測(cè)對(duì)象,如Class,Jsp,Servlet,Ejb等,Junit有不同的使用技巧,以后慢慢地分別講敘。以下以Class測(cè)試為例講解,除非特殊說(shuō)明。
2、下載安裝
去Junit主頁(yè)下載新版本3.8.1程序包junit-3.8.1.zip
用winzip或unzip將junit-3.8.1.zip解壓縮到某一目錄名為$JUNITHOME
將junit.jar和$JUNITHOME/junit加入到CLASSPATH中,加入后者只因?yàn)闇y(cè)試?yán)淘谀莻(gè)目錄下。
注意不要將junit.jar放在jdk的extension目錄下
運(yùn)行命令,結(jié)果如下圖。
java junit.swingui.TestRunner junit.samples.AllTests
3、Junit架構(gòu)
下面以Money這個(gè)類(lèi)為例進(jìn)行說(shuō)明。
public class Money {
private int fAmount;//余額
private String fCurrency;//貨幣類(lèi)型
public Money(int amount, String currency) {
fAmount= amount;
fCurrency= currency;
}
public int amount() {
return fAmount;
}
public String currency() {
return fCurrency;
}
public Money add(Money m) {//加錢(qián)
return new Money(amount()+m.amount(), currency());
}
public boolean equals(Object anObject) {//判斷錢(qián)數(shù)是否相等
if (anObject instanceof Money) {
Money aMoney= (Money)anObject;
return aMoney.currency().equals(currency())
&& amount() == aMoney.amount();
}
return false;
}
}
Junit本身是圍繞著兩個(gè)設(shè)計(jì)模式來(lái)設(shè)計(jì)的:命令模式和集成模式.
命令模式
利用TestCase定義一個(gè)子類(lèi),在這個(gè)子類(lèi)中生成一個(gè)被測(cè)試的對(duì)象,編寫(xiě)代碼檢測(cè)某個(gè)方法被調(diào)用后對(duì)象的狀態(tài)與預(yù)期的狀態(tài)是否一致,進(jìn)而斷言程序代碼有沒(méi)有bug。
當(dāng)這個(gè)子類(lèi)要測(cè)試不只一個(gè)方法的實(shí)現(xiàn)代碼時(shí),可以先建立測(cè)試基礎(chǔ),讓這些測(cè)試在同一個(gè)基礎(chǔ)上運(yùn)行,一方面可以減少每個(gè)測(cè)試的初始化,而且可以測(cè)試這些不同方法之間的聯(lián)系。
例如,我們要測(cè)試Money的Add方法,可以如下:
public class MoneyTest extends TestCase { //TestCase的子類(lèi)
public void testAdd() { //把測(cè)試代碼放在testAdd中
Money m12CHF= new Money(12, "CHF"); //本行和下一行進(jìn)行一些初始化
Money m14CHF= new Money(14, "CHF");
Money expected= new Money(26, "CHF");//預(yù)期的結(jié)果
Money result= m12CHF.add(m14CHF); //運(yùn)行被測(cè)試的方法
Assert.assertTrue(expected.equals(result)); //判斷運(yùn)行結(jié)果是否與預(yù)期的相同
}
}
如果測(cè)試一下equals方法,用類(lèi)似的代碼,如下:
public class MoneyTest extends TestCase { //TestCase的子類(lèi)
public void testEquals() { //把測(cè)試代碼放在testEquals中
Money m12CHF= new Money(12, "CHF"); //本行和下一行進(jìn)行一些初始化
Money m14CHF= new Money(14, "CHF");
Assert.assertTrue(!m12CHF.equals(null));//進(jìn)行不同情況的測(cè)試
Assert.assertEquals(m12CHF, m12CHF);
Assert.assertEquals(m12CHF, new Money(12, "CHF")); // (1)
Assert.assertTrue(!m12CHF.equals(m14CHF));
}
}
當(dāng)要同時(shí)進(jìn)行測(cè)試Add和equals方法時(shí),可以將它們的各自的初始化工作,合并到一起進(jìn)行,形成測(cè)試基礎(chǔ),用setUp初始化,用tearDown清除。如下:
public class MoneyTest extends TestCase {//TestCase的子類(lèi)
private Money f12CHF;//提取公用的對(duì)象
private Money f14CHF;
protected void setUp() {//初始化公用對(duì)象
f12CHF= new Money(12, "CHF");
f14CHF= new Money(14, "CHF");
}
public void testEquals() {//測(cè)試equals方法的正確性
Assert.assertTrue(!f12CHF.equals(null));
Assert.assertEquals(f12CHF, f12CHF);
Assert.assertEquals(f12CHF, new Money(12, "CHF"));
Assert.assertTrue(!f12CHF.equals(f14CHF));
}
public void testSimpleAdd() {//測(cè)試add方法的正確性
Money expected= new Money(26, "CHF");
Money result= f12CHF.add(f14CHF);
Assert.assertTrue(expected.equals(result));
}
}
將以上三個(gè)中的任一個(gè)TestCase子類(lèi)代碼保存到名為MoneyTest.java的文件里,并在文件首行增加
import junit.framework.*;
,都是可以運(yùn)行的。關(guān)于Junit運(yùn)行的問(wèn)題很有意思,下面單獨(dú)說(shuō)明。
上面為解釋概念“測(cè)試基礎(chǔ)(fixture)”,引入了兩個(gè)對(duì)兩個(gè)方法的測(cè)試。命令模式與集成模式的本質(zhì)區(qū)別是,前者一次只運(yùn)行一個(gè)測(cè)試。
集成模式
利用TestSuite可以將一個(gè)TestCase子類(lèi)中所有test***()方法包含進(jìn)來(lái)一起運(yùn)行,還可將TestSuite子類(lèi)也包含進(jìn)來(lái),從而行成了一種等級(jí)關(guān)系?梢园裈estSuite視為一個(gè)容器,可以盛放TestCase中的test***()方法,它自己也可以嵌套。這種體系架構(gòu),非常類(lèi)似于現(xiàn)實(shí)中程序一步步開(kāi)發(fā)一步步集成的現(xiàn)況。
對(duì)上面的例子,有代碼如下:
public class MoneyTest extends TestCase {//TestCase的子類(lèi)
....
public static Test suite() {//靜態(tài)Test
TestSuite suite= new TestSuite();//生成一個(gè)TestSuite
suite.addTest(new MoneyTest("testEquals")); //加入測(cè)試方法
suite.addTest(new MoneyTest("testSimpleAdd"));
return suite;
}
}
從Junit2.0開(kāi)始,有列簡(jiǎn)捷的方法:
public class MoneyTest extends TestCase {//TestCase的子類(lèi)
....
public static Test suite() {靜態(tài)Test
return new TestSuite(MoneyTest.class); //以類(lèi)為參數(shù)
}
}
TestSuite見(jiàn)嵌套的例子,在后面應(yīng)用案例中有。