什么是單元測(cè)試
首先介紹一下什么是單元測(cè)試漠畜,Wikipedia給出的定義是:在計(jì)算機(jī)編程中,單元測(cè)試是一種軟件測(cè)試方法坞靶,通過該方法憔狞,對(duì)單個(gè)源代碼單元,一套或多套計(jì)算機(jī)程序模塊以及相關(guān)的控制數(shù)據(jù)彰阴,使用過程和操作過程進(jìn)行測(cè)試瘾敢,以確定它們是否適合使用。
通俗來說硝枉,單元測(cè)試是為了測(cè)試某一個(gè)代碼單元而寫的測(cè)試代碼廉丽。但是什么叫“一個(gè)代碼單元”呢?是一個(gè)模塊妻味、還是一個(gè)類正压、還是一個(gè)方法(函數(shù))呢?不同的人责球、不同的語言焦履,都有不同的理解。一般的定義雏逾,尤其是是在OOP領(lǐng)域嘉裤,是一個(gè)類的一個(gè)方法。在此栖博,我們也這樣理解:?jiǎn)卧獪y(cè)試屑宠,是為了測(cè)試某一個(gè)類的某一個(gè)方法能否正常工作,而寫的測(cè)試代碼仇让。
如何進(jìn)行單元測(cè)試
一般來說典奉,我們寫單元測(cè)試躺翻,會(huì)用到一些單元測(cè)試框架。常見的Java單元測(cè)試框架有JUnit卫玖、TestNG等公你,在Android中會(huì)用到JUnit4、Mockito假瞬、Robolectric等陕靠。在這里,我們先通過JUnit4進(jìn)行簡(jiǎn)單單元測(cè)試編寫脱茉,關(guān)于JUnit4后面會(huì)有專門的文章進(jìn)行介紹剪芥,這里只是進(jìn)行單元測(cè)試演示。
針對(duì)某功能類進(jìn)行測(cè)試琴许,假設(shè)現(xiàn)在有個(gè)計(jì)算整數(shù)相加的類Calculator粗俱,如下:
public class Calculator {
public static int add(int a, int b) {
return a + b;
}
}
相信之前對(duì)沒有使用過單元測(cè)試的開發(fā)者來說,下面的代碼是測(cè)試此功能類的一種方式:
public class CalculatorTest {
public static void main(String[] args) {
int result = Calculator.add(1, 2);
System.out.println(result);
}
}
這樣沒錯(cuò)虚吟,但是如果要測(cè)試的方法很多,又或者結(jié)果互相影響签财,大量的測(cè)試類或者main方法集中大量測(cè)試方法串慰,造成項(xiàng)目代碼結(jié)構(gòu)以及可讀性降低,Android Studio 創(chuàng)建新項(xiàng)目默認(rèn)有專門的包結(jié)構(gòu)用來進(jìn)行單元測(cè)試唱蒸,這個(gè)往往被大部分沒有進(jìn)行單元測(cè)試的開發(fā)者忽略邦鲫。
androidTest
運(yùn)行在真機(jī)或者模擬器上,而不是運(yùn)行在本地環(huán)境下的虛擬機(jī)中神汹。如果在測(cè)試時(shí)庆捺,需要使用Android框架組件(例如Activity)、Android framework api 和 supporting api,或者Android特定的環(huán)境因素(例如Context),這時(shí)候你就要?jiǎng)?chuàng)建一個(gè)專門的測(cè)試類放置與此目錄下屁魏。
test
如果你的單元測(cè)試不依賴Android環(huán)境滔以,那么你需要使用一個(gè)本地的單元測(cè)試,運(yùn)行在本地的JVM上氓拼,這樣的話你画,就有利于你的測(cè)試速度,因?yàn)榧虞d物理設(shè)備是很耗費(fèi)時(shí)間的桃漾。
開始Unit Test
1.配置項(xiàng)目
在app/build.gradle中添加
testCompile ‘junit:junit:4.12’
2.生成測(cè)試類
代碼中選中需要測(cè)試的類坏匪,右鍵Go To —> Test選擇Create New Test
勾選需要測(cè)試的方法->OK
然后是提示選擇測(cè)試類需要放置的包文件夾,選擇test文件夾—>OK
這時(shí),我們會(huì)發(fā)現(xiàn)我們項(xiàng)目的test目錄下會(huì)多了一個(gè)和Calculator類相同包文件結(jié)構(gòu)的CalculatorTest測(cè)試類
當(dāng)然上面的這些創(chuàng)建測(cè)試類的步驟也可以手動(dòng)實(shí)現(xiàn)撬统,不過建議測(cè)試類的文件包結(jié)構(gòu)和需要測(cè)試的類相同适滓。
3.編寫測(cè)試代碼
public class CalculatorTest {
@Test
public void testAdd() {
int sum = Calculator.add(1, 2);
Assert.assertEquals(3, sum);
}
}
這里的CalculatorTest是Calculator對(duì)應(yīng)的測(cè)試類。而這里的testAdd()就是add()這個(gè)方法對(duì)應(yīng)的測(cè)試方法恋追。所以凭迹,寫單元測(cè)試罚屋,就是給你的每個(gè)類的每個(gè)public方法寫對(duì)于的測(cè)試方法。非public方法我們一般是不測(cè)試的蕊苗,雖然可以通過反射等手段去做沿后,但是一般看來,非public方法是這個(gè)類的實(shí)現(xiàn)細(xì)節(jié)朽砰,我們并不關(guān)心尖滚,我們只關(guān)心某一個(gè)public方法的輸入、輸出瞧柔。
一般來說漆弄,一個(gè)方法對(duì)應(yīng)的測(cè)試方法主要分為3部分,以上面的測(cè)試方法為例:
1.setup造锅。一般是new出你要測(cè)試的那個(gè)類撼唾,以及其他一些前提條件的設(shè)置:Calculator calculator = new Calculator();
2.執(zhí)行操作。一般是調(diào)用你要測(cè)試的那個(gè)方法哥蔚,獲得運(yùn)行結(jié)果:
int sum = calculator.add(1, 2);
3.驗(yàn)證結(jié)果倒谷。驗(yàn)證得到的結(jié)果跟預(yù)期中是一樣的:Assert.assertEquals(3, sum);
4.運(yùn)行單元測(cè)試
打開CalculatorTest,用鼠標(biāo)右鍵點(diǎn)擊testAdd()方法糙箍,選擇Run testAdd(),或者點(diǎn)擊testAdd()方法前的箭頭渤愁,運(yùn)行結(jié)果如下圖所示:
除了在AndroidStudio里面運(yùn)行,你還可以在命令行通過gradle testDebugUnitTest深夯,或者是gradle testReleaseUnitTest抖格,分別運(yùn)行debug和release版本的unit testing,這種方式可以一次性運(yùn)行所有測(cè)試類的所有測(cè)試方法咕晋。 運(yùn)行后雹拄,每個(gè)test case的報(bào)告可以在project_root/app/build/reports/tests/debug/index.html 這個(gè)xml里面看到。大致如下圖:
5.分析單元測(cè)試結(jié)果
上圖可以看到掌呜,測(cè)試方法執(zhí)行完后無異常信息顯示滓玖,且方法呈綠色表示測(cè)試預(yù)期與測(cè)試結(jié)果一致,測(cè)試通過质蕉。
另外還可能有兩種測(cè)試不通過情況發(fā)生:
1.代碼錯(cuò)誤呢撞,測(cè)試方法方法呈紅色,并報(bào)錯(cuò)相關(guān)代碼錯(cuò)誤信息饰剥,如圖:
2.測(cè)試結(jié)果與預(yù)期不一致殊霞,測(cè)試方法呈橙色,并展示測(cè)試代碼的預(yù)期結(jié)果與實(shí)際結(jié)果汰蓉,如圖:
小結(jié)
上面主要講述了單元測(cè)試的定義绷蹲,以及在android studio 上如何編寫單元測(cè)試類以及代碼,并分析單元測(cè)試結(jié)果的幾種情況。
后面我們會(huì)持續(xù)更新單元測(cè)試的必要性祝钢,使用Mockito等框架在Android上進(jìn)行單元測(cè)試的方法比规,以及對(duì)在實(shí)際項(xiàng)目中進(jìn)行單元測(cè)試的實(shí)踐。