平凡的故事:年輕開發(fā)者的那些傷心事
這并不是該年輕開發(fā)者的***工作經(jīng)歷。但是他的***個(gè)項(xiàng)目已經(jīng)被證明是有問題的。當(dāng)時(shí),他認(rèn)為功能從來不會(huì)改變。但是他錯(cuò)了,每次修改功能都需要一次徹底的重構(gòu),引發(fā)了bug以及大量時(shí)間的浪費(fèi)。他甚至嘗試過編寫測(cè)試之類的、有效的方式。不過他的測(cè)試需要維護(hù)、編寫所需時(shí)間、甚至更多執(zhí)行的時(shí)間。
對(duì)于每個(gè)年輕的開發(fā)者,他在成長(zhǎng)過程中總是聽到有經(jīng)驗(yàn)的開發(fā)者說到的“當(dāng)心!過早優(yōu)化是萬惡之源!”、“編寫測(cè)試!測(cè)試!測(cè)試!”。或許他只是在重構(gòu)一個(gè)微小的實(shí)用程序方法,而這時(shí)有經(jīng)驗(yàn)的開發(fā)者將走過來、以嚴(yán)厲的表情警告說“你沒有做過早優(yōu)化,對(duì)吧?”、或“你在寫測(cè)試,是嗎?”。
但是所有這些警告都被忽視了。因?yàn)槟贻p開發(fā)者不理解過早優(yōu)化為什么是萬惡之源、也不理解測(cè)試應(yīng)該是正確的。根據(jù)他自己的經(jīng)驗(yàn),他知道下面的需求從長(zhǎng)期看是不合理的(因?yàn)樗鼈兛赡苡行薷?,編寫測(cè)試就是在浪費(fèi)時(shí)間。
“我究竟為什么每次都必須重寫我的代碼?我究竟為什么必須在當(dāng)下編寫代碼、隨后再重構(gòu),我什么時(shí)候能夠編寫世界上***的代碼?還有,我究竟為什么不得不用我所有的時(shí)間來編寫沒用的測(cè)試?”這就是年輕開發(fā)者的疑惑。
一天,年輕開發(fā)者開始著手一個(gè)新項(xiàng)目。他決定無視有經(jīng)驗(yàn)開發(fā)者的警告;為了應(yīng)對(duì)每次需求變化,他期望每塊代碼是快速的、可配置的和健壯的。需求清晰了,不過他要做得更好。比如,當(dāng)有個(gè)功能,生成以大寫‘S’結(jié)尾的產(chǎn)品代碼時(shí),他創(chuàng)建一個(gè)配置對(duì)象,這樣結(jié)尾的字母就可以通過配置來修改,通過配置還可以決定這個(gè)字母應(yīng)該是大寫還是小寫。當(dāng)需求說明需要一些校驗(yàn)時(shí),他就創(chuàng)建一個(gè)龐大的校驗(yàn)器,不僅包含需求要求的,還有很多沒要求的。
在編寫了項(xiàng)目的核心之后,一種***的感覺充滿了年輕開發(fā)者的全身。“那個(gè)有經(jīng)驗(yàn)的開發(fā)者是錯(cuò)誤的!”年輕開發(fā)者看著自己的杰作得意地說。他夜以繼日地工作,認(rèn)為數(shù)周后就可以發(fā)布產(chǎn)品了。
光陰荏苒……
一天,客戶告知他們一個(gè)bug。有經(jīng)驗(yàn)的開發(fā)者看到這個(gè)bug,對(duì)顯示器上出現(xiàn)的情況保留著厭惡:年輕開發(fā)者看到了大教堂,而有經(jīng)驗(yàn)開發(fā)者看到的是貧民窟;年輕開發(fā)者看到了模式,而有經(jīng)驗(yàn)開發(fā)者看到的是一個(gè)充斥著class的復(fù)雜網(wǎng)絡(luò);年輕開發(fā)者看到了比光速還快的代碼,而有經(jīng)驗(yàn)開發(fā)者看到的是不必要的復(fù)雜算法。他不想碰這些代碼,因此他讓這個(gè)年輕的開發(fā)者去修復(fù)自己的bug。
其他人不認(rèn)為年輕開發(fā)者的代碼是優(yōu)美的,只有這個(gè)想法讓他感到失望。他充滿憤怒地打開了項(xiàng)目……才發(fā)現(xiàn)代碼對(duì)于他來說,也是費(fèi)解的!代碼背后沒有清晰的意義。“這就是我不打算再使用這門語言的原因,語法太糟糕了”,往往是年輕開發(fā)者的***反應(yīng)。但是他在內(nèi)心深處知道,這不是真正的問題。真正的問題是他。
一天結(jié)束的時(shí)候,bug修復(fù)好了,卻產(chǎn)生了另一個(gè)bug,這是那一天之后發(fā)現(xiàn)的。每次修復(fù)都在影響著項(xiàng)目?jī)?nèi)部的瘦弱的平衡,就像亮白色衣服上的一小塊黑色補(bǔ)丁。
此時(shí)這個(gè)年輕的開發(fā)者絕望了,他的大教堂開始搖晃,他感到離崩塌不遠(yuǎn)了。年輕開發(fā)者自問,“或許我不是這份工作的合適人選。為什么我不能編寫恰當(dāng)?shù)拇a呢?”帶著沮喪和憤怒的交織心情,年輕的開發(fā)者打開了有經(jīng)驗(yàn)開發(fā)者維護(hù)的項(xiàng)目。
他看到的代碼讓他感到吃驚:代碼有注釋和測(cè)試,易于閱讀。和他最初開始寫的代碼沒有太多區(qū)別,有一些清晰的例外:沒有可擴(kuò)展的配置,每行代碼都被測(cè)試了,每個(gè)方法都取著有意義的名字、且簡(jiǎn)短(最多10行代碼),只做必要的,每個(gè)文件只包含了能夠嚴(yán)格做本質(zhì)工作的方法。
在這個(gè)憂郁的時(shí)刻,有經(jīng)驗(yàn)開發(fā)者來到了年輕開發(fā)者身邊,和他挨得很近,開始重構(gòu)引起所有bug的代碼。
他們一起工作了數(shù)天,有時(shí)候,有經(jīng)驗(yàn)開發(fā)者寫代碼,而年輕開發(fā)者觀看有經(jīng)驗(yàn)開發(fā)者如何解決問題;另些場(chǎng)合,年輕開發(fā)者寫代碼,而有經(jīng)驗(yàn)開發(fā)者在旁邊監(jiān)督。
數(shù)天后,一次新的部署標(biāo)志著bug已被修復(fù)。引起bug的小部分代碼,現(xiàn)在可以被測(cè)試了、易于閱讀了,也很穩(wěn)定。有經(jīng)驗(yàn)開發(fā)者看著年輕開發(fā)者說:“你現(xiàn)在明白了嗎?”
年輕開發(fā)者點(diǎn)了點(diǎn)頭,他現(xiàn)在明白了。***的關(guān)鍵不是預(yù)測(cè)將來,而是編寫容易修改、測(cè)試(這樣修改就不會(huì)引發(fā)其它bug了)以及只需滿足當(dāng)前需求的代碼。當(dāng)他意識(shí)到這一點(diǎn)時(shí),他注意到他正在變化,正在變成差不多有經(jīng)驗(yàn)的開發(fā)者。
年輕開發(fā)者問,“我們現(xiàn)在能夠重構(gòu)整個(gè)項(xiàng)目嗎?”
有經(jīng)驗(yàn)開發(fā)者干脆地答道,“當(dāng)然不可以!沒有預(yù)算”
年輕開發(fā)者問,“但是,如果其它bug出現(xiàn)了,該怎么辦?”
有經(jīng)驗(yàn)開發(fā)者答道,“我們將找個(gè)自由職業(yè)者(freelancer)來修復(fù)”
然后,這個(gè)差不多有經(jīng)驗(yàn)的開發(fā)者開始編寫優(yōu)秀的代碼,準(zhǔn)備學(xué)習(xí)另外的經(jīng)驗(yàn)。不過這是另外一個(gè)故事了。
年輕開發(fā)者的啟示:回頭看看你過去寫的代碼,如果你的代碼看起來還不夠優(yōu)美,不要感到失望。
有經(jīng)驗(yàn)開發(fā)者的啟示:當(dāng)周圍有年輕開發(fā)者時(shí),你將不得不給他擦屁股。你***的機(jī)會(huì)就是他將學(xué)習(xí)如何編寫得體的代碼,越快越好。
自由職業(yè)者的啟示:你或許想提高你的報(bào)價(jià)
英文原文:The Sorrows of Young Developer
譯文: 《年輕開發(fā)者的那些傷心事 》 臘八粥