您的位置:軟件測(cè)試 > 開(kāi)源軟件測(cè)試 > 開(kāi)源單元測(cè)試工具 > junit
JUnit4概述(三)
作者:網(wǎng)絡(luò)轉(zhuǎn)載 發(fā)布時(shí)間:[ 2013/12/11 11:21:03 ] 推薦標(biāo)簽:

在Eclipse中使用JUnit4進(jìn)行單元測(cè)試(高級(jí)篇)

一、高級(jí)Fixture

上一篇文章中我們介紹了兩個(gè)Fixture標(biāo)注,分別是@Before和@After,我們來(lái)看看他們是否適合完成如下功能:有一個(gè)類是負(fù)責(zé)對(duì)大文件(超過(guò)500兆)進(jìn)行讀寫(xiě),他的每一個(gè)方法都是對(duì)文件進(jìn)行操作。換句話說(shuō),在調(diào)用每一個(gè)方法之前,我們都要打開(kāi)一個(gè)大文件并讀入文件內(nèi)容,這是一個(gè)非常耗費(fèi)時(shí)間的操作。如果我們使用@Before和@After,那么每次測(cè)試都要讀取一次文件,效率及其低下。這里我們所希望的是在所有測(cè)試一開(kāi)始讀一次文件,所有測(cè)試結(jié)束之后釋放文件,而不是每次測(cè)試都讀文件。JUnit的作者顯然也考慮到了這個(gè)問(wèn)題,它給出了@BeforeClass 和 @AfterClass兩個(gè)Fixture來(lái)幫我們實(shí)現(xiàn)這個(gè)功能。從名字上可以看出,用這兩個(gè)Fixture標(biāo)注的函數(shù),只在測(cè)試用例初始化時(shí)執(zhí)行@BeforeClass方法,當(dāng)所有測(cè)試執(zhí)行完畢之后,執(zhí)行@AfterClass進(jìn)行收尾工作。在這里要注意一下,每個(gè)測(cè)試類只能有一個(gè)方法被標(biāo)注為@BeforeClass 或 @AfterClass,并且該方法必須是Public和Static的。

二、測(cè)試。

還記得我在初級(jí)篇中給出的例子嗎,那個(gè)求平方根的函數(shù)有Bug,是個(gè)死循環(huán):

    public void squareRoot(int n) ...{

        for (; ;) ;                 //Bug : 死循環(huán)

    }

如果測(cè)試的時(shí)候遇到死循環(huán),你的臉上不會(huì)露出笑容。因此,對(duì)于那些邏輯很復(fù)雜,循環(huán)嵌套比較深的程序,很有可能出現(xiàn)死循環(huán),因此一定要采取一些預(yù)防措施。測(cè)試是一個(gè)很好的解決方案。我們給這些測(cè)試函數(shù)設(shè)定一個(gè)執(zhí)行時(shí)間,超過(guò)了這個(gè)時(shí)間,他們會(huì)被系統(tǒng)強(qiáng)行終止,并且系統(tǒng)還會(huì)向你匯報(bào)該函數(shù)結(jié)束的原因是因?yàn)槌瑫r(shí),這樣你可以發(fā)現(xiàn)這些Bug了。要實(shí)現(xiàn)這一功能,只需要給@Test標(biāo)注加一個(gè)參數(shù)即可,代碼如下:

@Test(timeout = 1000)

public void squareRoot() ...{

        calculator.squareRoot(4);

        assertEquals(2, calculator.getResult());

}


Timeout參數(shù)表明了你要設(shè)定的時(shí)間,單位為毫秒,因此1000代表1秒。

三、 測(cè)試異常

JAVA中的異常處理也是一個(gè)重點(diǎn),因此你經(jīng)常會(huì)編寫(xiě)一些需要拋出異常的函數(shù)。那么,如果你覺(jué)得一個(gè)函數(shù)應(yīng)該拋出異常,但是它沒(méi)拋出,這算不算Bug呢?這當(dāng)然是Bug,并JUnit也考慮到了這一點(diǎn),來(lái)幫助我們找到這種Bug。例如,我們寫(xiě)的計(jì)算器類有除法功能,如果除數(shù)是一個(gè)0,那么必然要拋出“除0異常”。因此,我們很有必要對(duì)這些進(jìn)行測(cè)試。代碼如下:

  @Test(expected = ArithmeticException.class)

  public void divideByZero() ...{

calculator.divide(0);

   }

如上述代碼所示,我們需要使用@Test標(biāo)注的expected屬性,將我們要檢驗(yàn)的異常傳遞給他,這樣JUnit框架能自動(dòng)幫我們檢測(cè)是否拋出了我們指定的異常。

四、     Runner (運(yùn)行器)

大家有沒(méi)有想過(guò)這個(gè)問(wèn)題,當(dāng)你把測(cè)試代碼提交給JUnit框架后,框架如何來(lái)運(yùn)行你的代碼呢?答案是——Runner。在JUnit中有很多個(gè)Runner,他們負(fù)責(zé)調(diào)用你的測(cè)試代碼,每一個(gè)Runner都有各自的特殊功能,你要根據(jù)需要選擇不同的Runner來(lái)運(yùn)行你的測(cè)試代碼?赡苣銜(huì)覺(jué)得奇怪,前面我們寫(xiě)了那么多測(cè)試,并沒(méi)有明確指定一個(gè)Runner?這是因?yàn)镴Unit中有一個(gè)默認(rèn)Runner,如果你沒(méi)有指定,那么系統(tǒng)自動(dòng)使用默認(rèn)Runner來(lái)運(yùn)行你的代碼。換句話說(shuō),下面兩段代碼含義是完全一樣的:

import org.junit.internal.runners.TestClassRunner;

import org.junit.runner.RunWith;

//使用了系統(tǒng)默認(rèn)的TestClassRunner,與下面代碼完全一樣

public class CalculatorTest ...{...}

@RunWith(TestClassRunner.class)

public class CalculatorTest ...{...}

從上述例子可以看出,要想指定一個(gè)Runner,需要使用@RunWith標(biāo)注,并且把你所指定的Runner作為參數(shù)傳遞給它。另外一個(gè)要注意的是,@RunWith是用來(lái)修飾類的,而不是用來(lái)修飾函數(shù)的。只要對(duì)一個(gè)類指定了Runner,那么這個(gè)類中的所有函數(shù)都被這個(gè)Runner來(lái)調(diào)用。后,不要忘了包含相應(yīng)的Package哦,上面的例子對(duì)這一點(diǎn)寫(xiě)的很清楚了。接下來(lái),我會(huì)向你們展示其他Runner的特有功能。

上一頁(yè)123456下一頁(yè)
軟件測(cè)試工具 | 聯(lián)系我們 | 投訴建議 | 誠(chéng)聘英才 | 申請(qǐng)使用列表 | 網(wǎng)站地圖
滬ICP備07036474 2003-2017 版權(quán)所有 上海澤眾軟件科技有限公司 Shanghai ZeZhong Software Co.,Ltd