Bug 少,性能好,容易修改。好的代碼影響深遠(yuǎn),而且它可能是產(chǎn)生10 倍工作效率的開發(fā)者的主要原因。盡管好代碼十分重要,但開發(fā)新手卻不得要領(lǐng)。關(guān)于這一主題的技巧多而冗雜,讓新手們?nèi)绾斡浀米。俊癈ode Complete(《代碼大全》)” 是這個(gè)主題的經(jīng)典,但內(nèi)容多達(dá) 960 頁!
我認(rèn)為應(yīng)該建立起良好的心態(tài),這樣,不管你用什么語言或者庫,都會(huì)自然而然的寫出高質(zhì)量的代碼。這里我主要談到 5 個(gè)相關(guān)的概念。記住它們,輕松寫出寫出好代碼。
請(qǐng)避免特立獨(dú)行
當(dāng)你讀到一些文章中的新技巧時(shí),如醍醐灌頂,一定會(huì)想要寫點(diǎn)看起來很聰明的代碼,讓同行們眼前一亮。
問題是人們只是想修完 BUG,然后繼續(xù)處理其它事情。那些聰明的技巧常常只會(huì)成為一種消遣。我曾經(jīng)在“將神經(jīng)科學(xué)應(yīng)用于軟件開發(fā)”中談到,當(dāng)人們被迫花心思來理解你的某段代碼時(shí),它們的“精神堆棧”會(huì)迅速填滿,因而難以理解其中深意。
[譯者注:圖片中的注釋內(nèi)容:這在 C 語言中用于避免誤寫成 variable = null。最近它造成不少人困惑,但似乎并沒帶來多大好處]
不要在工作中使用太多可能需要額外解釋的個(gè)性化方式。
不要用“你的方式”來編寫代碼,只需要按照標(biāo)準(zhǔn)(的代碼規(guī)范)來編寫就好。再次強(qiáng)調(diào),要寫讓人讀得明白,看得下去的代碼,讓人家能夠理解它。
分而治之
模塊化可以使復(fù)雜的代碼結(jié)構(gòu)變得清晰,除此之外還有很多方法可以達(dá)到同樣的目的,而無需創(chuàng)建更多函數(shù)。將長長的條件表達(dá)式保存為一到兩個(gè)變量就是個(gè)不錯(cuò)的方法,可以避免調(diào)用函數(shù)的開銷。這些變量可以用在其它地方,甚至可用于組合更復(fù)雜的條件。
拆解問題的方法在于盡可能的讓每個(gè)部分保持集中,只影響局部狀態(tài),不要混入不相關(guān)的問題,要避免副作用。編程語言和庫通常會(huì)帶來各自相應(yīng)的問題,避免這些問題可以讓你的代碼更專注于其表達(dá)的業(yè)務(wù)。單一責(zé)任原則就是通過集中代碼和局部化代碼帶來良好設(shè)計(jì)的例子。
[譯者注:圖中注釋內(nèi)容:這是不需要額外函數(shù)開銷的一種模塊化方法]
我喜歡利用變量來進(jìn)行邏輯劃分。
TDD(Test Driven Development,測(cè)試驅(qū)動(dòng)開發(fā))的成功實(shí)施表現(xiàn)出了它所帶來的好處,它迫使人們運(yùn)用一些以前不受歡迎的準(zhǔn)則。無狀態(tài)的代碼曾經(jīng)被嫌棄又慢又沒必要(大部分老的 C/C++ 代碼中可以看到),然而現(xiàn)在每個(gè)人都在談?wù)摷兒瘮?shù)。就算你不采用 TDD,你也應(yīng)該學(xué)習(xí)它背后的原則。在新的模式下工作會(huì)讓你成為適應(yīng)性極強(qiáng)的開發(fā)者。
分離代碼并使其可分別處理
你寫代碼的時(shí)候面臨著什么樣的困難,你的計(jì)算機(jī)和工具也面臨著同樣的困難。代碼的復(fù)雜性,與需要進(jìn)行的預(yù)處理和需要處理的突發(fā)情況存在著或多或少的聯(lián)系。
現(xiàn)在暫時(shí)拋開那些額外的構(gòu)建工具所帶來的好處。它們需要你使用特定領(lǐng)域的語言,比如自定義模板,或者復(fù)雜的動(dòng)態(tài)數(shù)據(jù)結(jié)構(gòu),比如哈稀表。IDE 通常不善于處理這些東西,要找到相關(guān)的代碼段則更加困難。
盡量避免使用不能很好支持 IDE 的語言擴(kuò)展和庫。它們給你的生產(chǎn)力帶來的好處,遠(yuǎn)大于簡易配置和用簡潔語法保存擊鍵帶來的小便利。
[譯者注:圖中注釋內(nèi)容:使用神奇的字體串可能造成 IDE 不能正確識(shí)別你的代碼]
ServiceLocator 是與 IDE 整合不佳的一個(gè)設(shè)計(jì)樣例。
另一個(gè)保持 IDE“整合度”的相關(guān)方法是避免編寫特殊的代碼。多數(shù)語言都提供了編寫動(dòng)態(tài)代碼的能力。如果濫用這些特性,比如特殊的字符串、特殊的數(shù)組索引和自定義模板語言特性等,會(huì)產(chǎn)生難以連接的代碼庫[譯者注:這里的連接應(yīng)該是指相互關(guān)聯(lián)的關(guān)系,連接最直接的影響是在使用 IDE 等工具進(jìn)行重構(gòu)的時(shí)候可以自動(dòng)根據(jù)連接關(guān)系修改相關(guān)引用]。一般說來,那些只有你一個(gè)人才能看懂的特性會(huì)讓你摔跟頭的,因?yàn)槿绻?IDE 不能理解這些代碼,在你想進(jìn)行結(jié)構(gòu)調(diào)整的時(shí)候,IDE 就沒法幫你進(jìn)行重構(gòu)。
讓程序可讀
致力于可預(yù)測(cè)的架構(gòu)。這種架構(gòu)下你的隊(duì)友要進(jìn)行某項(xiàng)查找就會(huì)很容易,可以節(jié)約不少時(shí)間。一旦你為項(xiàng)目確定了一個(gè)整體的架構(gòu),就一定要把主要元素放在顯眼的位置。使用 MVC?把模型、視圖和控制器放在他們自己的目錄下,不要放在三個(gè)深層次的目錄中,也不要放在幾個(gè)不同的地方。
我在前面談到了模塊化。也存在過度的模塊化,讓定位代碼這種事情變得艱難無比。IDE 可能會(huì)帶來一些幫助,但通常你往往會(huì)讓 IDE 忽略庫目錄,因?yàn)槠渲杏泻芏嗖幌嚓P(guān)的代碼,或者它的索引需要人工處理一些問題,就會(huì)造成兩敗俱傷的局面。盡量使用較少的庫,選用那些盡可能多覆蓋你需求的庫。
庫和工具也可能成為新人的障礙。我最近使用 EcmaScript 7 (babel) 構(gòu)建了一個(gè)項(xiàng)目,后來我才意識(shí)到我們的初級(jí)開發(fā)人員一直因?yàn)橄敫忝靼姿ㄔ谀抢铮@對(duì)團(tuán)隊(duì)的生產(chǎn)力造成了巨大損失。我低估了這對(duì)一個(gè)新手所帶來的壓力。不要使用對(duì)當(dāng)前來說太難掌握的工具,等時(shí)機(jī)成熟再使用。
這是我寫的一個(gè) makefile 中的真實(shí)代碼。新手不需掌握過多的新技術(shù)。
讓代碼易于理解
如果你已經(jīng)做到了這一點(diǎn),那我們來解決更重要的問題——選擇好名字,這是軟件開發(fā)中的重要部分。構(gòu)建工具在這方面不能提供幫助,原因很簡單,計(jì)算機(jī)不會(huì)真正知道解決方案背后的邏輯。你得通過文檔來解釋代碼,而與主題相關(guān),且符合上下文,體現(xiàn)變量和功能的名稱就能很好做到這一點(diǎn)。語義化的名稱甚至可以減少對(duì)文檔的需求。
在名稱中使用前綴對(duì)理解它們很有幫助。這在過去是一種流行的做法,我認(rèn)為對(duì)這種作法的誤用導(dǎo)致了它的消亡。像匈牙利命名法這樣的前綴系統(tǒng)最初只是為了增加意義,但最后用于其中的上下文越來越少,終于少得只剩類型信息。
[譯者注:圖中的注釋內(nèi)容:使用名稱來表達(dá)意圖,不要利用語言來耍小聰明]
近來,F(xiàn)luent 接口經(jīng)常被濫用。
最后要說說老生常談的回溯復(fù)雜度。簡單地說就是要盡可能減少條件分支的數(shù)目。每多一個(gè)分支都會(huì)增加縮進(jìn),同時(shí)降低可讀性。不過更重要的是,增加的東西越多,你需要跟蹤的東西就越多。
結(jié)論
本文介紹了五個(gè)簡單的總體概念,我希望你能從中輕松的學(xué)習(xí)到組織代碼的方法。
實(shí)踐是最好的老師,用編程來鞏固理論。如果你還沒有這樣做,我誠摯向你推薦代碼大全。它帶來了大量的示例,幾乎剖析了你可能遇到的每一種問題。
-
代碼
+關(guān)注
關(guān)注
30文章
4880瀏覽量
70026
原文標(biāo)題:編寫良好的代碼:如何減少代碼的認(rèn)知負(fù)荷
文章出處:【微信號(hào):machinelearningai,微信公眾號(hào):機(jī)器學(xué)習(xí)算法與人工智能】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
何為高質(zhì)量的代碼?如何寫出高質(zhì)量代碼?

【明天會(huì)更好】程序員如何在面試時(shí)寫出高水平代碼
FPGA寫出來的代碼質(zhì)量差
用verilog寫出代碼
這么秀的代碼,你能寫出來嗎?
為什么有時(shí)候會(huì)寫出爛代碼

一本教你怎么寫出讓同事無法維護(hù)的代碼
如何寫出高效優(yōu)美的C語言代碼

如何寫出穩(wěn)定的單片機(jī)代碼

評(píng)論