1.Given-When-Then分段
每個case其實(shí)都可以分為三步走,1.mock對象孽亲,準(zhǔn)備測試數(shù)據(jù)坎穿。2.調(diào)用目標(biāo)API 3.驗(yàn)證輸出和行為。所以我們可以用如下方式將3步分別放入Given-When-Then三個分段中。(為了保密玲昧,代碼做了改動)
- (void)testNeedToShowRowWhenTypeIsAll{
//given _sut.collectionType = 1;
NSIndexPath *path1 = [NSIndexPath indexPathForRow:1 inSection:1];
//when
BOOL needToShow1 = [_sut needToShowRow:path1]; //then
assertThatBool(needToShow1,isTrue());}
這樣我們一眼掃過去就可以清晰的看出一個case大體上都在干什么栖茉。
2.一個Case只測試一種情況
可能我們調(diào)用的一個API內(nèi)部有一個if…else…。我建議if一個case孵延,else一個case吕漂。分兩個不同的case來作測試.這樣每個case就很清晰自己在測試什么東西。而如果全部雜糅在一個case中尘应,可讀性會降低不少痰娱,而且case體積也會變得相對大很多,因?yàn)槟阋狦iven-When-Then兩次菩收。更不建議在case中寫for循環(huán)驗(yàn)證梨睁。有人說我的測試目標(biāo)函數(shù)中有很多if…else…,那么我覺得你應(yīng)該重構(gòu)下你的設(shè)計(jì)了。
所以娜饵,我們的結(jié)論是一個Case只測試一種情況坡贺,不同情況用When標(biāo)明:
- (void)testNeedToShowRowWhenTypeIsAll{
...
}
- (void)testNeedToShowRowWhenTypeIsOnSell{
...
}
3.用_sut來標(biāo)明被測試類
一個測試文件只有一個被測試類。但是當(dāng)我們的測試文件越來越多的時候箱舞,當(dāng)我們看一個測試case的時候需要看懂這個case才明白我們的被測試類是誰遍坟」Γ或者我們也可以看測試文件名(XXXXXXTest.m)才知道我們的被測試類是誰隆判,但是這樣卻不是很直觀询枚。所以不管我們在那個測試文件中狐肢,測試的類是誰舱殿,叫什么名字狡门,我們都以為一個局部變量名_sut來定義我們的被測試類值纱。這樣我們一眼就能知道我們被測試類是誰今穿。
_sut就是System Under Test的縮寫寂呛。
@implementation JHSCollectionDataSourceTest { JHSCollectionDataSource *_sut;
}
- (void)testNeedToShowHeaderWhenTypeIsAll{
//given
_sut.collectionType = 1;
//when
BOOL needToShow1 = [_sut needToShowHeader:1]; //then
assertThatBool(needToShow1,isTrue());
}
4.用Category暴露私有函數(shù)和屬性
我們的測試case中調(diào)用的方法可能會改變一個私有的屬性怎诫,調(diào)用一個私有的方法。怎么去優(yōu)雅的驗(yàn)證這種行為呢贷痪,我們可以在測試文件的開頭用一個名字為UnitTest的category來暴露出我們的私有方法和屬性(屬性暴露的是屬性對應(yīng)的getter和setter方法)幻妓。
@interface JHSTestDataSource (UnitTest)
- (NSInteger)getSellGroupCount;
- (BOOL)needShowHeader:(NSInteger)section;
@end