如何在JavaScript中謹(jǐn)慎使用代碼注釋
前言
最令程序員頭痛的事情莫過于閱讀別人的代碼,但其實時間一久閱讀自己的代碼也會很痛苦。 問題不是出在『自己或別人』,而是在代碼本身。
必要的注釋可以闡明實現(xiàn)細(xì)節(jié)和設(shè)計意圖,以此節(jié)約自己和別人的時間。 然而很多時候注釋起的作用卻適得其反,比如自動生成的過多的注釋分散閱讀者的注意力, 而過期的失效的注釋更是誤導(dǎo)閱讀者。
自動生成的注釋
代碼注釋的泛濫想必是從Eclipse,Visual Studio等IDE開始的。 這些IDE提供了很多快捷功能,生成類/接口的骨架,具有Getter/Setter的屬性等等。 如果用過IDE,下面的代碼你一定不會陌生:
/** * @param args */ public static void main(String[] args) { // TODO Auto-generated method stub }
上述6行代碼中的4行注釋包含的信息量是0,既沒有闡釋參數(shù)args是何物,也沒有說明main的用途。 然而大量的項目中都充斥著這樣的自動生成注釋。
『建議』:如果有參數(shù)或機(jī)制需要說明,請補充這些信息。否則請刪除自動生成注釋。 當(dāng)然,用于生成文檔的注釋除外。
過多的注釋
總會有人不厭其煩地編寫長篇累牘的注釋,或無微不至,或語焉不詳,或晦澀難懂,或文采飛揚。 總之沒有幫助我更快閱讀代碼的注釋都是失敗的注釋。
為了說明問題,Harttle克隆了4.x Linux Kernel源碼, 來大致分析一下其注釋行數(shù)。 我們知道內(nèi)核代碼95%以上是C語言,所以統(tǒng)計.c文件就足夠說明問題了。
➜ linux git:(master) git clone git@github.com:torvalds/linux.git --depth=1 ➜ linux git:(master) find . -name "*.c" -o -name "*.h" -exec grep -E '^\s*((\*)|(/[/*]))' {} \; | wc -l 724804 ➜ linux git:(master) find . -name "*.c" -o -name "*.h" -exec cat {} \; | wc -l 4018961 ➜ linux git:(master) node > 724804/(4018961-724804) 0.22002715717556875
內(nèi)核倉庫中的代碼大概是402萬行(未移除空行),其中注釋72萬行,占比22%。 Linux內(nèi)核使用低級的C語言編寫,涉及到復(fù)雜的CPU調(diào)度、內(nèi)存管理,驅(qū)動程序。 因此注釋會偏多一些,一般的項目注釋應(yīng)小于這個數(shù)值。
『建議』:如果你的代碼中注釋超過了20%,那么顯然你過度注釋了。
文件頭注釋
很多編輯器/IDE都會生成默認(rèn)的文件頭,例如:
/** * @file /tmp/xxx.js * @author harttle(yangjvn@126.com) * @date 2016-08-30 22:33 * @description A XXX Implementation for XXX. */
文件頭注釋清晰地列出了文件的作者、功能描述等信息,看起來很有用。 不過這樣的文件頭存在的問題在于其維護(hù)性:
- 其他人做小的修改時未必會修改@author,甚至連@author都不知道現(xiàn)在該文件已經(jīng)面目全非。
- 每次移動該文件,是否還需要花功夫更新 @file 信息?
- 誰會在每次代碼修改后記得更新 @description,于是@description也總是誤導(dǎo)讀者
文件頭注釋意在維護(hù)代碼文件的元信息,以便在分發(fā)和部署過程中維護(hù)作者版權(quán)等信息。 然而在擁有版本控制的代碼倉庫中,這些信息不再需要手動維護(hù),甚至可以通過git blame查看每一行代碼的作者和時間信息。
『建議』:使用版本控制工具,刪除文件頭注釋。版權(quán)信息可在構(gòu)建或分發(fā)時生成。
冗余的注釋
意圖非常清楚的代碼原則上不需要注釋,多余的注釋反而會造成維護(hù)性問題。 尤其是非英語母語的作者常常會掉到這個坑里。比如變量和函數(shù)的注釋:
/* * 獲取用戶數(shù)目 */ function getUserCount(){ // 用戶的列表 var userList = []; }
這不是廢話么!冗余的注釋問題仍然在于維護(hù)性,例如調(diào)整函數(shù)功能、調(diào)整參數(shù)順序, 或者更換變量名時我們不得不更新這些注釋。否則這些注釋就會誤導(dǎo)下一個讀者。
【建議】:不說廢話。
抽取注釋到標(biāo)識符
可能讀者也會有這樣的經(jīng)驗:當(dāng)我們寫了一大段代碼時,往往需要把它們分為幾塊。 然后在每一塊開頭添加一段注釋。例如:
function calcTotalCharge(movies, user){ // Calculate Movie Charge var movieCharge = 0; for(var i=0; i<movies.length; i++){ var charge = 0; if(movie.type === 'discount'){ charge = movie.charge * 0.8; } else if(movie.type === 'short'){ charge = movie.charge * 2; } else if(movie.type === 'normal'){ charge = movie.charge; } movieCharge += charge; } // Calculate User Charge var rentCharge = 0; if(user.isVIP1){ rentCharge = 10; } if(user.isVIP2){ rentCharge = 200; } else if(user.isVIP3){ rentCharge = 300; } else if(user.isVIP4){ rentCharge = 500; } // Calculate Total Charge return movieCharge + rentCharge; }
上述代碼中的三段注釋確實加速了閱讀代碼的速度, 但每當(dāng)代碼需要注釋才能讀懂時就應(yīng)該警醒:是不是結(jié)構(gòu)設(shè)計有問題。 對于上述代碼,我們可以通過更加可復(fù)用的結(jié)構(gòu)來消除注釋:
function calcTotalCharge(movies, user){ return calcMovieCharge(movies) + calcUserCharge(user); } function calcMovieCharge(movies){ var total = 0; for(var i=0; i<movies.length; i++){ total += calcSingleMovieCharge(movie); } return total; } function calcSingleMovieCharge(movie){ if(movie.type === 'discount') return movie.charge * 0.8; else if(movie.type === 'short') return movie.charge * 2; else if(movie.type === 'normal') return movie.charge; return 0; } function calcUserCharge(user){ if(user.isVIP1) return 10; else if(user.isVIP2) return 200; else if(user.isVIP3) return 300; else if(user.isVIP4) return 500; return 0; }
代碼重構(gòu)之后原來的注釋就變得毫無意義,代碼意圖都被清晰的表述在標(biāo)識符的命名中。 通常重構(gòu)會帶來代碼量的減小,因為封裝了分支、每個單元的邏輯也更加明確。
【建議】:當(dāng)我們發(fā)現(xiàn)不得不進(jìn)行注釋時,需要警醒是否結(jié)構(gòu)設(shè)計發(fā)生了問題。
有用的注釋
至此Harttle已描述了這么多反模式,并非為了說明代碼注釋不重要。 而是為了說明『代碼注釋存在的意義在于幫助理解代碼本身』。 例如在編寫一些Trick,Polyfill,臨時代碼,以及復(fù)雜算法時,注釋變得相當(dāng)重要。 例如:
- Tricks and Polyfills。有時簡單的Trick就可解決多數(shù)問題問題, 為沒必要編寫復(fù)雜的普適算法, 例如檢測瀏覽器的DOM API支持,檢測AMD/CommonJS環(huán)境等等。 這時我們需要清晰地說明這些Trick的意圖,甚至可以將這些代碼抽離為polyfill模塊。
- 復(fù)雜算法。有時我們會編寫數(shù)學(xué)性非常強的算法,一眼望去不知所云。 在開始這些算法前清晰地說明其意圖何在,讀者也就不必花大功夫讀懂這些數(shù)學(xué)了。
- 公有接口。模塊的對外接口從邏輯上定義了模塊類型,公有接口代碼也更容易被人讀到。 尤其是JavaScript接口:如果不注釋options中到底是什么,誰曉得接口如何使用。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
微信小程序canvas開發(fā)水果老虎機(jī)的思路詳解
這篇文章主要介紹了微信小程序canvas開發(fā)水果老虎機(jī)的思路,本文通過思路代碼分步驟給大家介紹的非常詳細(xì),具有一定的參考借鑒價值,需要的朋友可以參考下2020-02-02