近在測試結(jié)算時用到JUNIT框架,發(fā)現(xiàn)JUNIT4有一些新特性很好用,現(xiàn)分享給大家。
一、 @BeforeClass 和 @AfterClass
我們經(jīng)常用到的兩個Fixture標(biāo)注(@Before和@After)是在每個測試用例前、后執(zhí)行,現(xiàn)新增了一種特性是在所有測試一開始讀一次文件,所有測試結(jié)束之后釋放文件,而不是每次測試都讀文件。從名字上可以看出,用這兩個Fixture標(biāo)注的函數(shù),只在測試用例初始化時執(zhí)行 @BeforeClass方法,當(dāng)所有測試執(zhí)行完畢之后,執(zhí)行@AfterClass進(jìn)行收尾工作。在這里要注意一下,每個測試類只能有一個方法被標(biāo)注為 @BeforeClass 或 @AfterClass,并且該方法必須是Public和Static的。
二、 測試
對于那些邏輯很復(fù)雜,循環(huán)嵌套比較深的程序,很有可能出現(xiàn)死循環(huán),因此一定要采取一些預(yù)防措施。測試是一個很好的解決方案。我們給這些測試函數(shù)設(shè)定一個執(zhí)行時間,超過了這個時間,他們會被系統(tǒng)強(qiáng)行終止,并且系統(tǒng)還會向你匯報(bào)該函數(shù)結(jié)束的原因是因?yàn)槌瑫r,這樣你可以發(fā)現(xiàn)這些Bug了。要實(shí)現(xiàn)這一功能,只需要給@Test標(biāo)注加一個參數(shù)即可,代碼如下:
@Test(timeout = 1000)
public voidsquareRoot() ...{
calculator.squareRoot(4);
assertEquals(2, calculator.getResult());
}
Timeout參數(shù)表明了你要設(shè)定的時間,單位為毫秒,因此1000代表1秒。
三、 測試異常
JAVA中的異常處理也是一個重點(diǎn),因此你經(jīng)常會編寫一些需要拋出異常的函數(shù)。那么,如果你覺得一個函數(shù)應(yīng)該拋出異常,但是它沒拋出,這算不算Bug呢?這當(dāng)然是Bug,并 JUnit也考慮到了這一點(diǎn),來幫助我們找到這種Bug。例如,我們寫的計(jì)算器類有除法功能,如果除數(shù)是一個0,那么必然要拋出“除0異常”。因此,我們很有必要對這些進(jìn)行測試。代碼如下:
@Test(expected = ArithmeticException.class)
public void divideByZero() ...{
calculator.divide(0);
}
如上述代碼所示,我們需要使用@Test標(biāo)注的expected屬性,將我們要檢驗(yàn)的異常傳遞給他,這樣JUnit框架能自動幫我們檢測是否拋出了我們指定的異常。
四、 Runner (運(yùn)行器)
在JUnit中有很多個 Runner,他們負(fù)責(zé)調(diào)用你的測試代碼,每一個Runner都有各自的特殊功能,你要根據(jù)需要選擇不同的Runner來運(yùn)行你的測試代碼?赡苣銜X得奇怪,前面我們寫了那么多測試,并沒有明確指定一個Runner啊?這是因?yàn)镴Unit中有一個默認(rèn)Runner,如果你沒有指定,那么系統(tǒng)自動使用默認(rèn)Runner來運(yùn)行你的代碼。換句話說,下面兩段代碼含義是完全一樣的:
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 ...{
...
}
從上述例子可以看出,要想指定一個Runner,需要使用@RunWith標(biāo)注,并且把你所指定的Runner作為參數(shù)傳遞給它。另外一個要注意的是,@RunWith是用來修飾類的,而不是用來修飾函數(shù)的。只要對一個類指定了Runner,那么這個類中的所有函數(shù)都被這個Runner來調(diào)用。后,不要忘了包含相應(yīng)的Package哦,上面的例子對這一點(diǎn)寫的很清楚了。
原文鏈接:http://testing.etao.com/