這是我第二次讀《測(cè)試驅(qū)動(dòng)開(kāi)發(fā)》這本書翁逞,第一次讀的時(shí)候,我剛剛從TWU回來(lái)屏鳍,書里有著太多我剛剛學(xué)到的東西勘纯,我看完了之后很神奇的,沒(méi)有對(duì)TDD有更深的感悟钓瞭,反而對(duì)JAVA的繼承有了一些新的理解驳遵。由于十一之前出了一段時(shí)間的bug,凱峰老師推薦我再去讀一讀這本書山涡,讀完之后堤结,對(duì)TDD有了一些感悟。
首先先簡(jiǎn)單的介紹一下這本書中主要的概念:測(cè)試驅(qū)動(dòng)開(kāi)發(fā)鸭丛。
測(cè)試驅(qū)動(dòng)開(kāi)發(fā)
這里的測(cè)試這個(gè)概念主要是指自動(dòng)化測(cè)試中的單元測(cè)試竞穷。單元測(cè)試的意義在于,它可以快速的運(yùn)行鳞溉,使開(kāi)發(fā)可以快速的獲得反饋瘾带。而且單元測(cè)試越多,覆蓋的功能越全熟菲,可以極大的增強(qiáng)開(kāi)發(fā)對(duì)于代碼的信心看政。測(cè)試驅(qū)動(dòng)開(kāi)發(fā)是以測(cè)試為優(yōu)先,先進(jìn)行測(cè)試代碼的編寫抄罕,以通過(guò)測(cè)試為目的進(jìn)行開(kāi)發(fā)允蚣。
測(cè)試驅(qū)動(dòng)開(kāi)發(fā)這件事,在我接觸這件事之前呆贿,是完全無(wú)法想象的嚷兔。在我學(xué)編程的時(shí)候,我從來(lái)沒(méi)有想過(guò)寫測(cè)試榨崩,編寫代碼更多的時(shí)候是一件一氣呵成的事谴垫,但是之后手動(dòng)測(cè)試的時(shí)候?qū)映霾桓F的bug、反復(fù)的調(diào)試是一件極其痛苦的事情母蛛◆婕簦花一晚上寫代碼,但是花兩天的調(diào)試簡(jiǎn)直是常事彩郊。
在我接觸到測(cè)試驅(qū)動(dòng)開(kāi)發(fā)了之后前弯,很多時(shí)候也是補(bǔ)測(cè)試蚪缀,就是倒過(guò)來(lái)的DDT(Development Driven Test)。這是一件很奇怪的事恕出,在TWU的時(shí)候询枚,大家學(xué)習(xí)TDD,將DDT當(dāng)作一個(gè)笑話講出來(lái)浙巫,但是回到了工作中金蜀,DDT確是常態(tài)。有的時(shí)候我甚至是恐懼寫測(cè)試的畴。
《測(cè)試驅(qū)動(dòng)開(kāi)發(fā)》這本書里很常見(jiàn)的一句話是渊抄,測(cè)試可以增強(qiáng)你的信心。在我沒(méi)接觸過(guò)自動(dòng)化測(cè)試甚至是接觸了一段時(shí)間內(nèi)丧裁,我都不知道測(cè)試的意義是什么护桦,只是大家說(shuō)要這樣做,我就這樣做煎娇。但是有一天二庵,我參加一個(gè)比賽,礙于時(shí)間缓呛,我沒(méi)有寫測(cè)試催享,這讓我想起了曾經(jīng)被調(diào)試支配的恐懼。那個(gè)時(shí)候我無(wú)比的懷念每天陪伴我的這些測(cè)試强经。
單就測(cè)試而不是驅(qū)動(dòng)開(kāi)發(fā)來(lái)說(shuō)睡陪,就是讓重構(gòu)變成一件簡(jiǎn)單的事寺渗。每當(dāng)我做了一點(diǎn)改變的時(shí)候匿情,我很恐懼引入bug,所以我跑了一遍測(cè)試信殊,測(cè)試完好地通過(guò)了炬称,我就會(huì)稍微安心一些的進(jìn)行接下來(lái)的工作。所以那些沒(méi)有被測(cè)試覆蓋的功能讓我恐懼涡拘,所以每當(dāng)我不想寫測(cè)試時(shí)玲躯,回想一下那些反反復(fù)復(fù)的調(diào)試,我會(huì)覺(jué)得還是寫測(cè)試讓我更加舒服鳄乏。
說(shuō)回測(cè)試驅(qū)動(dòng)開(kāi)發(fā)跷车,我有一個(gè)很神奇的毛病,當(dāng)我看一本書的時(shí)候橱野,這里邊的概念會(huì)在一定的時(shí)間的深深地影響到我朽缴。所以在我看《測(cè)試驅(qū)動(dòng)開(kāi)發(fā)》的這段日子,我迎來(lái)了一個(gè)新功能的代碼水援,新功能不復(fù)雜密强,復(fù)雜的是需要與之前的代碼進(jìn)行集成茅郎,而且之前的代碼還很多,有點(diǎn)亂或渤。我腦子中有一點(diǎn)想法系冗,但是又不知道如何入手。
所以我理所當(dāng)然的想到了先寫一個(gè)簡(jiǎn)單的測(cè)試薪鹦。
我要做的新功能是這樣的掌敬,我需要調(diào)用一個(gè)api,這個(gè)api會(huì)返回給我一系列的response池磁,我根據(jù)這些response中的某一個(gè)字段來(lái)進(jìn)行狀態(tài)判斷并且存到數(shù)據(jù)庫(kù)中涝开。所以對(duì)于我來(lái)說(shuō),最簡(jiǎn)單的場(chǎng)景就是框仔,我調(diào)用api失敗了舀武,我沒(méi)有得到期望的response,我應(yīng)該存一個(gè)有問(wèn)題的狀態(tài)進(jìn)數(shù)據(jù)庫(kù)离斩。我只需要添加一小點(diǎn)代碼银舱,這個(gè)測(cè)試就順利的通過(guò)了。這之后我寫了一個(gè)最常見(jiàn)的happy path的測(cè)試場(chǎng)景:我成功的獲取了response跛梗,并且response的值都是我期待的值寻馏,我應(yīng)該存儲(chǔ)一個(gè)好的狀態(tài)進(jìn)入數(shù)據(jù)庫(kù)。根據(jù)這個(gè)測(cè)試核偿,我加入了一定的邏輯進(jìn)去诚欠。這之后我又列舉了各種bad path作為測(cè)試場(chǎng)景,這些場(chǎng)景驅(qū)動(dòng)我完善了我的代碼漾岳。代碼完成之后的使用真實(shí)數(shù)據(jù)的手動(dòng)測(cè)試也非常的順利轰绵,時(shí)間在我看來(lái)花費(fèi)的比想象的要少,而且完成后沒(méi)有很燒腦很累的感覺(jué)尼荆。
在我完成這個(gè)之后左腔,再回過(guò)頭來(lái)看TDD,才開(kāi)始明白TDD對(duì)于一個(gè)開(kāi)發(fā)來(lái)說(shuō)它的魅力何在捅儒,我也對(duì)以通過(guò)測(cè)試為目的進(jìn)行開(kāi)發(fā)有了一點(diǎn)新的理解液样。對(duì)于我來(lái)說(shuō),一個(gè)功能有各種場(chǎng)景巧还,一開(kāi)始就想全想好是一個(gè)很燒腦的過(guò)程鞭莽,在寫好代碼之后回憶代碼都實(shí)現(xiàn)了哪些場(chǎng)景,然后編寫單元測(cè)試也很累麸祷,而且還可能會(huì)漏掉某些場(chǎng)景澎怒。但是先寫好測(cè)試場(chǎng)景,是一個(gè)很好的讓我梳理應(yīng)用場(chǎng)景的過(guò)程摇锋,并且可以很好的引導(dǎo)我的思維丹拯。每添加一個(gè)測(cè)試站超,我要做的就只是做一點(diǎn)改變,然后讓這個(gè)測(cè)試通過(guò)即可乖酬,每次我只需要關(guān)注很小的一部分就夠了死相。
讓一切從測(cè)試開(kāi)始,這是一個(gè)很好的實(shí)踐咬像。