軟件設計的哲學:第三章 編程的戰術和戰略

做者簡介: 常柱,微信公衆號【架構將來】做者,十多年一線互聯網研發從業經驗;前五八同城商業會員技術負責人,寶駕租車技術總監,現58 到家業務中臺技術負責人。程序員

 

 

 

 

好的軟件設計最重要的元素之一是在處理編程任務時採用的思惟方式。許多組織鼓勵一種戰術心態,專一於讓特性儘量快地工做。然而,若是你想要一個好的設計,你必須採起一種更有策略的方法,投入時間來產生乾淨的設計並解決問題。本章討論了爲何戰略方法能夠產生更好的設計,而且從長遠來看實際上比戰術方法更便宜。編程

3.1 戰術的編程

大多數程序員使用一種我稱爲戰術編程的思惟方式來進行軟件開發。在戰術方法中,您的主要關注點是讓某些東西工做起來,好比一個新特性或一個 bug 修復。乍一看,這彷佛是徹底合理的:還有什麼比編寫有效的代碼更重要呢?然而,戰術規劃使它幾乎不可能產生一個良好的系統設計。微信

戰術編程的問題在於它是短視的。 若是你在戰術上編程,你是在試圖儘快完成一項任務。也許你有一個艱難的最後期限。所以,規劃將來並非優先事項。你不會花不少時間去尋找最好的設計;你只是想盡快開始工做。你告訴本身,若是可讓當前的任務更快完成,增長一點複雜性或者引入一兩個小的封裝是能夠的。架構

這就是系統變得複雜的緣由。如前一章所述,複雜性是遞增的。使一個系統變得複雜的不是某一件特定的事情,而是數十個或數百個小事情的積累。若是你有策略地編程,每一個編程任務都會增長一些複雜性。爲了快速完成當前的任務,它們中的每個彷佛都是合理的妥協。然而,複雜性迅速增長,特別是若是每一個人都在戰術上編程。tornado

不久以後,一些複雜的問題就會開始產生,你會開始後悔當初沒有走那些捷徑。可是,您會告訴本身,讓下一個功能正常工做比返回並重構現有代碼更重要。從長遠來看,重構可能會有所幫助,但它確定會下降當前任務的速度。所以,您須要尋找快速補丁來解決遇到的任何問題。這隻會產生更多的複雜性,這就須要更多的補丁。很快,代碼就亂成一團了,但到目前爲止,狀況已經很是糟糕,須要幾個月的時間才能清理乾淨。你的時間表不可能容忍這樣的延遲,並且解決一兩個問題彷佛也不會有太大的不一樣,因此你只是在戰術上繼續編程。學習

若是您已經在一個大型軟件項目中工做了很長時間,我懷疑您已經在工做中看到過戰術編程,並體驗過它所致使的問題。一旦你開始走上戰術道路,就很難改變。測試

幾乎每一個軟件開發組織都至少有一個將戰術編程發揮到極致的開發人員:戰術旋風。戰術旋風是一個多產的程序員誰泵出的代碼比別人快得多,但工做在一個徹底戰術的方式。 當涉及到實現快速特性時,沒有人比戰術性 tornado 完成得更快。在一些組織中,管理層將戰術旋風視爲英雄。然而,戰術旋風留下了破壞的尾跡。他們不多被未來必須使用他們的代碼的工程師視爲英雄。一般,其餘工程師必須清理戰術旋風留下的混亂,這使得那些工程師(真正的英雄)看起來比戰術旋風進展緩慢。翻譯

3.2 戰略規劃

成爲一名優秀的軟件設計師的第一步是認識到僅僅爲了完成工做編寫代碼是不夠的。爲了更快地完成當前的任務而引入沒必要要的複雜性是不可接受的。最重要的是這個系統的長期結構。 任何系統中的大多數代碼都是經過擴展示有的代碼庫來編寫的,所以做爲開發人員,您最重要的工做就是促進這些將來的擴展。所以,您不該該認爲「工做代碼」是您的主要目標,儘管您的代碼固然必須工做。您的主要目標必須是產生一個偉大的設計,這也碰巧工做。這是戰略規劃。設計

戰略規劃須要一種投資心態。 您必須投入時間來改進系統的設計,而不是以最快的方式來完成當前的項目。這些投資在短時間內會讓你慢下來一點,但在長期內會讓你加快速度,如圖 3.1 所示。blog

一些投資將是積極的例如,花一點額外的時間爲每一個新類找到一個簡單的設計是值得的;與其實施第一個出如今腦海中的想法,不如嘗試幾個替代的設計,選擇最乾淨的一個。試着想象一些系統在將來可能須要改變的方式,並確保你的設計是簡單的。編寫好的文檔是主動投資的另外一個例子。

其餘投資將是被動的。 不管您預先投入多少,在您的設計決策中都會不可避免地出現錯誤。隨着時間的推移,這些錯誤將變得顯而易見。當你發現一個設計問題時,不要忽視它或修補它;花一點額外的時間來修復它。若是您有策略地進行編程,您將不斷地對系統設計進行小的改進。這與戰術編程相反,在戰術編程中,您不斷地添加小的複雜性,這些複雜性會在未來致使問題。

3.3 投資多少?

那麼,正確的投資額是多少呢?一筆巨大的前期投資,好比試圖設計整個系統,是不會有效的。這就是瀑布法,咱們知道它行不通。隨着您對系統的經驗的積累,理想的設計每每會零零碎碎地出現。所以,最好的方法是在連續的基礎上進行大量的小額投資。 我建議您將總開發時間的 10-20%用於投資。這個量足夠小,不會對您的日程安排產生重大影響,可是足夠大,隨着時間的推移會產生顯著的好處。所以,您最初的項目將比純戰術方法多花費 10-20%的時間。這些額外的時間將致使更好的軟件設計,而且您將在幾個月內開始體驗這些好處。用不了多久,你的開發速度就會比戰術編程至少快 10-20%。在這一點上,你的投資是免費的:從你過去的投資中得到的收益將節省足夠的時間來彌補將來投資的成本。你將很快收回最初投資的成本。圖 3.1 說明了這種現象。

圖 3.1:在開始階段,一種戰術的編程方法將比一種戰略方法更快地取得進展。然而,在戰術方法下,複雜性積累得更快,從而下降了生產率。隨着時間的推移,戰略方法取得了更大的進展。注意:這個數字只是一個定性的說明;我不知道任何經驗測量的準確曲線形狀。

相反,若是你有策略地進行編程,你將會以 10-20%的速度完成你的第一個項目,可是隨着時間的推移,你的開發速度會隨着複雜性的增長而減慢。用不了多久,您的編程速度至少會下降 10-20%。您將很快地歸還您在開始時節省的全部時間,而且在剩下的系統生命週期中,您將比採用策略方法時開發得更慢。若是您從未在嚴重降級的代碼庫中工做過,請與曾經工做過的人交談;他們會告訴你,糟糕的代碼質量至少會下降 20%的開發速度。

3.4 創業與投資

在某些環境中,有強大的力量反對戰略方法。例如,處於早期階段的初創公司會感到巨大的壓力,要求他們儘快發佈本身的早期版本。在這些公司,彷佛 10-20%的投資都是負擔不起的。所以,許多初創公司採起戰術性的方法,在設計上花費的精力不多,在出現問題時進行清理的時間更少。他們認爲這樣作是合理的,若是他們成功了,他們就會有足夠的錢聘請更多的工程師來清理垃圾。

若是您所在的公司傾向於這個方向,那麼您應該意識到,一旦代碼庫變成了意大利麪條,就幾乎不可能修復了。您可能要爲產品的生命週期支付高昂的開發成本。此外,好的(或壞的)設計的回報來得很快,因此頗有可能戰術方法甚至不會加快您的第一個產品發佈。

另外一件須要考慮的事情是,公司成功最重要的因素之一是工程師的素質。下降開發成本的最佳方法是僱傭優秀的工程師:他們的成本並不比平庸的工程師高多少,但卻擁有驚人的高生產率。 然而,最好的工程師都很是關心好的設計。若是你的代碼庫是一個殘骸,消息會傳出去,這將使你更難招募。結果,你極可能以平庸的工程師而了結。這將增長您將來的成本,並可能致使系統結構進一步退化。

Facebook 就是一個鼓勵戰術編程的初創公司。多年來,公司的座右銘是「快速行動,打破常規」。「剛從大學畢業的新工程師被鼓勵當即投入到公司的代碼庫中;對於工程師來講,在工做的第一週將提交推動到生產中是很正常的。從積極的一面來看,Facebook 創建了一個賦予員工權力的公司聲譽。工程師有很大的自由度,幾乎沒有什麼規則和限制來阻礙他們。

Facebook 做爲一家公司已經取得了驚人的成功,但它的代碼基礎卻由於公司的戰術方法而受損;不少代碼都不穩定,很難理解,只有不多的註釋和測試,並且很難處理。隨着時間的推移,公司意識到它的文化是不可持續的。最終,Facebook 改變了它的座右銘,「以堅實的基礎設施快速發展」,以鼓勵它的工程師在好的設計上投入更多。Facebook 可否成功地解決多年戰術編程積累的問題仍有待觀察。

公平地說,我應該指出,Facebook 的代碼可能並不比創業公司的平均水平差多少。戰術編程在初創公司中很常見;Facebook 就是一個特別明顯的例子。

幸運的是,在硅谷,戰略方法也有可能得到成功。谷歌和 VMware 與 Facebook 差很少是在同一時期成長起來的,但這兩家公司都採起了更具戰略性的策略。兩家公司都很是重視高質量的代碼和良好的設計,而且都構建了可以用可靠的軟件系統解決複雜問題的複雜產品。兩家公司強大的技術文化在硅谷廣爲人知。不多有公司能與他們競爭頂尖的技術人才。

這些例子代表,一個公司能夠用任何一種方法取得成功。然而,在一家關心軟件設計並擁有乾淨代碼庫的公司工做要有趣得多。

3.5 結論

好的設計不是免費的。它必須是你不斷投資的東西,這樣小問題就不會積累成大問題。 幸運的是,好的設計最終會收回成本,並且比你想象的要快。

在應用戰略方法時保持一致是相當重要的,要把投資看做是今天要作的事情,而不是明天要作的事情。當你陷入困境時,很容易把清理工做推遲到危機結束後。然而,這是一個滑坡;在當前的危機以後,幾乎能夠確定會有另外一個危機,在那以後又會有一個。一旦你開始延遲設計改進,延遲就很容易變成永久性的,你的文化也很容易滑入戰術方法。解決設計問題的時間越長,問題就越大;解決方案變得更加使人生畏,這使得人們更容易把它們推遲。最有效的方法是讓每一個工程師持續地對好的設計進行小的投資。


免責聲明:本文翻譯僅供學習使用,本文的版權歸英文原做者或出版方,如有侵權,請聯繫刪除。

相關文章
相關標籤/搜索