合格的Java程序測試究竟是怎樣的

作為測試驅(qū)動(dòng)設(shè)計(jì)和開發(fā)的忠實(shí)粉絲缕溉,我相信創(chuàng)造良好的測試是我們作為Java開發(fā)人員可以做的最重要的事情之一。我們寫測試出于許多原因:

塑造系統(tǒng)的設(shè)計(jì)瓣俯。我們知道輸入和輸出應(yīng)該是什么樣的传轰,但是我們需要?jiǎng)?chuàng)建什么對象來做到這一點(diǎn)呢?代碼應(yīng)該塑造成什么樣的“形狀”甘萧?編寫測試可以讓我們知道應(yīng)該創(chuàng)建什么樣的代碼。

為了確保初始和持續(xù)的正確性梆掸。讓我們的應(yīng)用程序如期望地那樣運(yùn)作并且始終如一地精確很重要扬卷。測試應(yīng)該竭力確保做到這一點(diǎn)。

文檔酸钦。測試是系統(tǒng)的文檔怪得,因?yàn)樗鼤f明它應(yīng)該做什么以及應(yīng)該怎么做。

那么“好的測試”到底是什么樣子的呢卑硫?

給測試命名

測試的名字至關(guān)重要徒恋,特別是從文檔角度來看的話。我們應(yīng)該能夠大聲讀出測試的名字就像一組需求一樣欢伏。事實(shí)上入挣,有一個(gè)偉大的IntelliJ插件,叫Enso颜懊,它會將你的測試名轉(zhuǎn)變?yōu)榍『梦挥诿總€(gè)類旁邊的語句财岔,這樣你就可以明明白白地看到你在做什么风皿。

不要以“test”開始命名測試的名稱河爹。這是來自于JUnit初期的后遺癥,當(dāng)需要它執(zhí)行的時(shí)候桐款。你的Test類將在Test文件夾中咸这,在一個(gè)最后有Test這個(gè)單詞的類中。會有一個(gè)@Test的注解魔眨。我們知道這是一個(gè)測試媳维。

你也應(yīng)該避免以“should”或“will”開頭。這些都是干擾詞遏暴。既然你已經(jīng)為這個(gè)功能寫了一個(gè)測試侄刽,那我們就知道它“should或will”工作(如果不能工作的話,那我們知道我們需要修復(fù)它)朋凉。

將測試名稱當(dāng)作一個(gè)要求州丹。 下面是一些例子:

addingNumbersWillSumValuesTogether()explodesOnNegativeID()notifiesListenersOnUpdates()

不要害怕表達(dá)出來。如果你的測試名稱確實(shí)需要很長的一串單詞,那就這么做墓毒,只要它能清楚說明將發(fā)生什么事情吓揪。

測試代碼

測試將分為3個(gè)部分:設(shè)置,操作所计,斷言柠辞。

設(shè)置

對你的測試設(shè)置代碼應(yīng)該只與在測試中被斷言的值相關(guān)。如果你有多余的設(shè)置代碼主胧,那就會搞不清楚它是什么叭首,并且與測試不相關(guān)。

這可以通過多種方式實(shí)現(xiàn):

將通用設(shè)置移動(dòng)到使用@Before注解的具體設(shè)置方法讥裤。

將重復(fù)的設(shè)置代碼移動(dòng)到輔助方法

使用Maker來創(chuàng)建復(fù)雜的測試對象放棒,并只設(shè)置測試中相關(guān)的值。

我重申一下:每個(gè)測試的設(shè)置部分應(yīng)該只有與最后被斷言的值相關(guān)的代碼己英。

不好的例子:

@TestpublicvoidreturnsBooksWherePartialTitleMatchesInAnyCast(){? ? ? ? Bookstore bookstore =newBookstore();? ? ? ? Book harryPotterOne =newBook("Harry Potter and The Philosopher Stone");? ? ? ? bookstore.add(harryPotterOne);? ? ? ? bookstore.add(newBook("Guardians of the Galaxy"));? ? ? ? Book harryPotterTwo =newBook("The Truth about HARRY POTTER");? ? ? ? bookstore.add(harryPotterTwo);? ? ? ? List results = bookstore.findByTitle("RY pot");? ? ? ? assertThat(results.size(), is(2));? ? ? ? assertThat(results, containsInAnyOrder(harryPotterOne, harryPotterTwo));? ? }

書店的初始化發(fā)生在測試中间螟,書本的創(chuàng)建也是。這讓測試顯得混亂不堪损肛,讓人搞不清楚發(fā)生了什么事情厢破。

好的例子:

privateBookstore bookstore =newBookstore();privateBook aHarryPotterBook =newBook("Harry Potter and The Philosopher Stone");privateBook anotherHarryPotterBook =newBook("The Truth about HARRY POTTER");privateBook aBook =newBook("Guardians of the Galaxy");@TestpublicvoidreturnsBooksWherePartialTitleMatchesInAnyCast(){bookstore.add(aHarryPotterBook);bookstore.add(aBook);bookstore.add(anotherHarryPotterBook);List results = bookstore.findByTitle("RY pot");assertThat(results.size(), is(2));assertThat(results, containsInAnyOrder(aHarryPotterBook, anotherHarryPotterBook));}

初始化發(fā)生在字段中,這樣在測試中發(fā)生了什么一清二楚治拿。

操作:

小菜一碟摩泪!最好保持到一行,你要進(jìn)行測試的獨(dú)立操作劫谅。有時(shí)候见坑,你專門測試的是輸出是什么,如果某些東西被多次調(diào)用捏检,或者在某些優(yōu)先操作之后調(diào)用的結(jié)果是什么荞驴,所以這不是一個(gè)硬性規(guī)定。當(dāng)讀取測試時(shí)贯城,用戶應(yīng)該快速而輕松地能說“將這些值設(shè)置成這樣熊楼,如果我執(zhí)行這個(gè)操作/這些操作,那么這是預(yù)期的結(jié)果”能犯。在上面的例子中鲫骗,便是bookstore.findByTitle()方法。

斷言:

使用Hamcrest踩晶。 Hamcrest是一個(gè)很棒的庫执泰,給我們一個(gè)流暢的API用來寫入測試。不會像這樣的代碼:

assertEquals(results.size(), 2);assertTrue(results.contains(aHarryPotterBook))assertTrue(results.contains(anotherHarryPotterBook))

我們可以一目了然渡蜻、輕松地閱讀像這樣的代碼:

assertThat(results.size(),is(2));assertThat(results, containsInAnyOrder(aHarryPotterBook, anotherHarryPotterBook));

這些相當(dāng)簡單的例子:Hamcrest有很多偉大的方法术吝,使編寫復(fù)雜測試變得很容易,并允許你創(chuàng)建自己的匹配器。

當(dāng)然顿苇,理想情況下峭咒,我們希望有一個(gè)獨(dú)立的斷言。這可以讓我們知道我們正在測試什么纪岁,并說明我們的代碼沒有意外情況凑队。就像這篇文章中所說的那樣,這不是一個(gè)硬性的規(guī)則幔翰,因?yàn)樵谀承┣闆r下漩氨,這是必要的,但如果你有這樣一個(gè)的測試:

assertThat(orderBook.bids.size(),is(4));assertThat(orderBook.asks.size(),is(3));assertThat(orderBook.bids.get(0).price,is(5200));assertThat(orderBook.asks.get(2).price,is(10000000));assertThat(orderBook.asks.get(2).isBuy,is(false));

那么要理解測試哪里失敗或哪條斷言重要就變得困難多了遗增。

你也可以在Hamcrest中編寫自定義的匹配器叫惊,因?yàn)镠amcrest可為復(fù)雜斷言提供一個(gè)優(yōu)雅的解決方案。如果你需要在一個(gè)循環(huán)中運(yùn)行斷言做修,或者你有大量的字段要斷言霍狰,那么一個(gè)自定義的匹配器可能才是上上之選。

一個(gè)測試的最重要的部分之一是饰及,當(dāng)它失敗時(shí)蔗坯,哪怕是一個(gè)5歲孩子也應(yīng)該看得出什么地方出了錯(cuò)以及哪里錯(cuò)了。失敗的消息一定不能含糊燎含。關(guān)于這方面的解決方法是:

如果做任何類型的對象比較宾濒,那么保證對象有一個(gè)體面的toString()消息。沒有什么比<MyObject @ 142131>不匹配更糟的了屏箍。

想要做的更好的話绘梦,可以對你的對象使用自定義匹配器。你可以準(zhǔn)確地知道哪些字段未能匹配赴魁。

確保明確為什么你要選擇和這個(gè)值作比較卸奉。例如,如果你正在將一個(gè)字段值與數(shù)字3000比較尚粘,那么為什么是3000择卦?你應(yīng)該費(fèi)力地明白這一點(diǎn)敲长。顯然郎嫁,這個(gè)數(shù)字不是隨便得來的,并且還要確保該變量的命名可以顯示它的值是如何得來的祈噪。

所有這些都應(yīng)該是在一個(gè)適度的常識范圍內(nèi)泽铛。沒有嚴(yán)格規(guī)定!

點(diǎn)贊+關(guān)注然后私信回復(fù)我“666”獲取免費(fèi)的全套Java學(xué)習(xí)資料

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末辑鲤,一起剝皮案震驚了整個(gè)濱河市盔腔,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖弛随,帶你破解...
    沈念sama閱讀 216,496評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件瓢喉,死亡現(xiàn)場離奇詭異,居然都是意外死亡舀透,警方通過查閱死者的電腦和手機(jī)栓票,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,407評論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來愕够,“玉大人走贪,你說我怎么就攤上這事』蟀牛” “怎么了坠狡?”我有些...
    開封第一講書人閱讀 162,632評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長遂跟。 經(jīng)常有香客問我逃沿,道長,這世上最難降的妖魔是什么幻锁? 我笑而不...
    開封第一講書人閱讀 58,180評論 1 292
  • 正文 為了忘掉前任感挥,我火速辦了婚禮,結(jié)果婚禮上越败,老公的妹妹穿的比我還像新娘触幼。我一直安慰自己,他們只是感情好究飞,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,198評論 6 388
  • 文/花漫 我一把揭開白布置谦。 她就那樣靜靜地躺著,像睡著了一般亿傅。 火紅的嫁衣襯著肌膚如雪媒峡。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,165評論 1 299
  • 那天葵擎,我揣著相機(jī)與錄音谅阿,去河邊找鬼。 笑死酬滤,一個(gè)胖子當(dāng)著我的面吹牛签餐,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播盯串,決...
    沈念sama閱讀 40,052評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼氯檐,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了体捏?” 一聲冷哼從身側(cè)響起冠摄,我...
    開封第一講書人閱讀 38,910評論 0 274
  • 序言:老撾萬榮一對情侶失蹤糯崎,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后河泳,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體沃呢,經(jīng)...
    沈念sama閱讀 45,324評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,542評論 2 332
  • 正文 我和宋清朗相戀三年拆挥,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了樟插。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,711評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡竿刁,死狀恐怖黄锤,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情食拜,我是刑警寧澤鸵熟,帶...
    沈念sama閱讀 35,424評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站负甸,受9級特大地震影響流强,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜呻待,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,017評論 3 326
  • 文/蒙蒙 一打月、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧蚕捉,春花似錦奏篙、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,668評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至敛熬,卻和暖如春肺稀,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背应民。 一陣腳步聲響...
    開封第一講書人閱讀 32,823評論 1 269
  • 我被黑心中介騙來泰國打工话原, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人诲锹。 一個(gè)月前我還...
    沈念sama閱讀 47,722評論 2 368
  • 正文 我出身青樓繁仁,卻偏偏與公主長得像,于是被迫代替她去往敵國和親辕狰。 傳聞我的和親對象是個(gè)殘疾皇子改备,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,611評論 2 353

推薦閱讀更多精彩內(nèi)容