請停止寫“爛”代碼!
譯文【51CTO.com快譯】作為一名在校教師,我的工作就是不斷審查學(xué)生們編寫的代碼。
這里我不想取笑任何人,因此所有實(shí)例皆以匿名形式公布,單純希望大家能夠從中獲得啟示。閑言少敘,咱們先來看一個奇妙的例子:
if(a<8&&b<8&&c<8)
a = 8;b=8;c=8;
大家不妨看看其中的錯誤在哪。
怎么樣,大家發(fā)現(xiàn)了多少問題?
我們先對其進(jìn)行分析。最惹眼的肯定是“if”的密度。其中的三項(xiàng)條件之間沒有用空格隔開。由于&運(yùn)算符跟8相鄰,因此很容易讓人看花眼。這就是問題之一,空格。大家***在代碼內(nèi)容間加上空格,包括各語句之間、運(yùn)算符之前與之后。下面我們對其進(jìn)行重寫:
if(a < 8 && b < 8 && c < 8)
很明顯,現(xiàn)在看起來好多了。
問題二
有過編程經(jīng)驗(yàn)的朋友對“魔數(shù)”肯定有著深刻印象。魔數(shù)是指那些直接出現(xiàn)在代碼當(dāng)中,不會自我描述且對于閱讀者而言毫無意義的數(shù)字。如果大家對于“8”這個數(shù)字沒有什么特別的負(fù)面感覺,那可以肯定您的編程從業(yè)時間并不長。這里歡迎大家來猜猜這個數(shù)字的含義。由于正是我本人給學(xué)生們布置的作業(yè),所以我很清楚這里“8”的意義(代表的是工人下班的最早時間),但對他人來說則很難猜。更糟糕的是,數(shù)字“8”在這段代碼中重復(fù)編寫了6次,意味著如果我們要將其換成7,則需要多次修改并反復(fù)查看以確認(rèn)沒有漏掉某個“8”。
解決方案當(dāng)然是使用常量。這里我們可以定義一條常量:
finalint EARLIEST_HOUR = 8;
現(xiàn)在代碼變成了這樣:
if(a < EARLIEST_HOUR && b < EARLIEST_HOUR && c < EARLIEST_HOUR)
a = EARLIEST_HOUR; b = EARLIEST_HOUR; c = EARLIEST_HOUR;
接下來看看代碼中的語義:這段代碼到底要實(shí)現(xiàn)什么效果?現(xiàn)在有了常數(shù),看起來應(yīng)該更為清楚。“if”負(fù)責(zé)測試三個變量,a、b與c。如果三個變量的值全面小于8,則全部獲得最早時間值分配。這類測試非常常見且能夠防止非法輸入狀況。然而,讓我們再來看看if本體——如果條件為真,哪行代碼會執(zhí)行if結(jié)果?
在開始之前,我們首先要強(qiáng)調(diào)一點(diǎn),Java是會直接忽略空格的。這意味著就編譯器而言,我們可以隨意使用空格但并非強(qiáng)制性(當(dāng)然,我們必須至少在兩個詞之間添加一個空格,但僅此而已)。這同時意味著Java中的語句會在編譯器發(fā)現(xiàn)“;”符號時終止,而非在行末自動終止?,F(xiàn)在再來看看以上代碼。其全部寫在一行當(dāng)中。由于這位同學(xué)在行前加了個“tab”,因此可以確定這一行屬于“if”語句。不過問題仍然存在——if的本體究竟是什么?以下為答案,if的本體是:
if(a < EARLIEST_HOUR && b < EARLIEST_HOUR && c < EARLIEST_HOUR)
a = EARLIEST_HOUR; b = EARLIEST_HOUR; c = EARLIEST_HOUR;
就是這樣,只有“a”屬于條件本體。那么語句剩下的部分呢?它們對于“if”毫無意義,意味著其無論條件是否為真都會被執(zhí)行。如果這位學(xué)生認(rèn)為條件為真時三項(xiàng)分配會被全部執(zhí)行,那他就犯下了一個嚴(yán)重的邏輯錯誤——代碼不可能實(shí)現(xiàn)這樣的效果,但卻很難看得出現(xiàn)。
另外,我們還發(fā)現(xiàn)了另外兩個問題:
問題三——條件為真時只有一次分配能夠切實(shí)執(zhí)行,而這與程序員的預(yù)期不符。
問題四——將太多語句寫進(jìn)同一行是種壞習(xí)慣。這并不是錯誤,Java允許這種作法,但其會干擾我們的審查與判斷。
下面來看正確的代碼編寫方式:
if(a < EARLIEST_HOUR && b < EARLIEST_HOUR && c < EARLIEST_HOUR) {
a = EARLIEST_HOUR;
b = EARLIEST_HOUR;
c = EARLIEST_HOUR;
}
總體來講——這里列舉的代碼示例非常糟糕,其中包含大量問題且令程序員很難發(fā)現(xiàn)其中的錯誤與bug。我的建議時,作為剛剛?cè)腴T的程序員,大家應(yīng)當(dāng)編寫結(jié)構(gòu)更為清晰且簡潔的代碼、正確使用空格、常量與縮進(jìn)等元素。
下面我們再來聊幾個大家可能感興趣的后續(xù)問題:
· 在***的代碼示例中,我利用該條件的本體包含了三個語句。我們該如何繼續(xù)使用單行語句編寫三次分配?如果不使用括號,又該如何實(shí)現(xiàn)?
· 我提到“8”是個無法解釋的“魔數(shù)”,但我并沒有提到該變量的名稱——a、b與c。您怎么看?這種命名方式是否也有問題?
· 如果a、b或者c中僅有一個變量的值小于8,您認(rèn)為結(jié)果會如何?
原文鏈接:https://dzone.com/articles/bad-bad-code
原文標(biāo)題:Bad, Bad Code! (And Tips to Avoid It)
原文作者:Shay Tavor
【51CTO譯稿,合作站點(diǎn)轉(zhuǎn)載請注明原文譯者和出處為51CTO.com】