一、前言
我們在學校學習或者有參加過C語言培訓的話,應該都聽說過“不建議使用goto
語句”。但是,一般不會有人告訴你為什么不建議使用goto
語句,類似于這種存在但不建議使用的關(guān)鍵詞還有很多。
今天,我們就一起來看看,那些在C語言中存在但又不建議使用的關(guān)鍵詞!
二、慎用goto關(guān)鍵詞
關(guān)于goto
語句的爭議已經(jīng)不是一天兩天了,大部分C語言老師在講到goto
這關(guān)鍵字的時候一般都會叫大家慎用goto
關(guān)鍵字。既然goto
語句C語言標準中定義了,那為什么不建議使用呢?
因為goto
語句不僅讓代碼的可讀性很差,隨意的跳出還會給程序帶來安全隱患。但正是這種幾乎被各大公司明令禁止使用的語句,在Linux內(nèi)核中卻被大量使用著。
這只能說明一點,那就是因為我們水平太菜了,公司怕因為你的一句goto
造成代碼莫名跑飛!
?
早期的程序員用goto
來解決代碼無法預料的后果,遇到什么問題就用一句goto
,讓程序跳轉(zhuǎn)到某個指定語句。
但是如果你的水平不夠,不能完全理解整個代碼的執(zhí)行過程的話,貿(mào)然使用goto
就可以出現(xiàn)莫名的問題,并且程序還很難被查到!
不建議使用goto
語句的原因還有以下幾點原因:
-
goto
語句可以被結(jié)構(gòu)化程序的別的語句代替; -
goto
語句會導致程序可讀性下降,因為在實際程序中,goto
可以跳到任何地方,可以往前可以往后,看程序慢慢看,看到goto
然后又要去找標識符到底跳到了哪里,可讀性嚴重下降,讓讀程序的人很不舒服; -
調(diào)試不舒服,調(diào)試程序時,由于有
goto
亂跳的,這就很難調(diào)試,去掉嘛,要重新寫代碼,不去掉,無從下手;
存在即合理,goto
語句也不例外,goto
它存在,確實在某些程序中使用可能有好處,但在我們學習的階段,應該盡量不要碰這類程序,養(yǎng)成一個好的編程習慣。記住一句話:**別人寫的goto我能看懂,但是我自己不會去寫goto!
**
三、慎用extern關(guān)鍵詞
在C語言程序中,我們用extern
關(guān)鍵字對某個變量作 “外部變量申明” ,表示該變量是一個已經(jīng)定義的外部變量,編譯器就會自動地在所有源文件里面查找該變量的定義。
但是在公司編程規(guī)范中有明確要求:不允許在C文件中使用 “extern” 來申明外部函數(shù)或全局變量
具體原因如下(其中一點原因):
這樣使用extern
來定義全局變量確實能給我們帶來了很大的便利,從而節(jié)省了我們很多的時間和精力。但是這樣做也會存在一些危險,比如我們在c3.c
文件引用的在a1.c
文件的funca
函數(shù)原型由UINT funca(UINT uiValue)
變?yōu)?code style="font-size:14px;padding:2px 4px;margin:0 2px;background-color:rgba(27,31,35,.05);font-family:'Operator Mono', Consolas, Monaco, Menlo, monospace;color:rgb(239,112,96);">UINT funca (UINT uiValue1, UINT uiValue2)我們在編譯的時候不會報錯,但是在我們執(zhí)行程序的時候會在使用該函數(shù)的時候存在危險,尤其是該函數(shù)若有一個參數(shù)為指針,極有可能會存在對指針的誤操作,而引起異常。
嵌入式特別是單片機的程序,最易犯的錯誤是全局變量滿天飛。此現(xiàn)象在早期匯編轉(zhuǎn)型過來的程序員以及初學者中常見,這幫家伙幾乎把全局變量當作函數(shù)形參來用。
每當看到這種程序,我總要戚眉變臉而后拍桌怒喝。沒錯,就是怒喝,我不否認全局變量的重要性,但我認為要十分謹慎地使用它,濫用全局變量會引申帶來其它更為嚴重的結(jié)構(gòu)性系統(tǒng)問題。
-
濫用全局變量會造成不必要的常量頻繁使用,特別當這個常量沒有用宏定義“正名”時,代碼閱讀起來將萬分吃力。
-
會導致軟件分層的不合理,全局變量相當于一條快捷通道,它容易使程序員模糊了“設(shè)備層”和“應用層”之間的邊界。寫出來的底層程序容易自作多情地關(guān)注起上層的應用。這在軟件系統(tǒng)的構(gòu)建初期的確效率很高,功能調(diào)試進度一日千里,但到了后期往往bug一堆,處處“補丁”,雷區(qū)遍布。說是度日如年舉步維艱也不為過。
-
由于軟件的分層不合理,到了后期維護,哪怕僅是增加修改刪除小功能,往往要從上到下掘地三尺地修改,涉及大多數(shù)模塊,而原有的代碼注釋卻忘了更新修改,這個時候,交給后來維護者的系統(tǒng)會越來越像一個“泥潭”,注釋的唯一作用只是使泥潭上方再加一些烏煙瘴氣。
-
全局變量大量使用,少不了有些變量流連忘返于中斷與主回圈程序之間。這個時候如果處理不當,系統(tǒng)的bug就是隨機出現(xiàn)的,無規(guī)律的,這時候初步顯示出病入膏肓的特征來了,沒有大牛來力挽狂瀾,注定慢性死亡。
無需多言,您已經(jīng)成功得到一個畸形的系統(tǒng),它處于一個神秘的穩(wěn)定狀態(tài)!你看著這臺機器,機器也看著你,相對無言,心中發(fā)毛。你不確定它什么時候會崩潰,也不曉得下一次投訴什么時候道理。
然后,我告訴大家現(xiàn)實層面的后果是什么。
-
“老人”氣昂昂,因為系統(tǒng)離不開他,所有“雷區(qū)”只有他了然于心。當出現(xiàn)緊急的bug時,只有他能夠搞定。你不但不能辭退他,還要給他加薪。
-
新人見光死,但凡招聘來維護這個系統(tǒng)的,除了改出更多的bug外,基本上一個月內(nèi)就走人,到了外面還宣揚這個公司的軟件質(zhì)量有夠差夠爛。
-
隨著產(chǎn)品的后續(xù)升級,幾個月沒有接觸這個系統(tǒng)的原創(chuàng)者會發(fā)現(xiàn),很多雷區(qū)他本人也忘記了,于是每次的產(chǎn)品升級維護周期越來越長,因為修改一個功能會冒出很多bug,而按下一個bug,會彈出其他更多的bug。在這期間,又會產(chǎn)生更多的全局變量。終于有一天他告訴老板,不行啦不行啦,資源不夠了,ram或者flash空間太小了,升級升級。
-
客戶投訴不斷,售后也快崩潰了,業(yè)務員也不敢推薦此產(chǎn)品了,市場份額越來越小,公司形象越來越糟糕。
注:以上關(guān)于extern的建議來源于黃工的分享。
四、慎用指針
指針對于初學者來說本來就是一個不易理解的東西,初學者一般都不能夠真正的理解指針,并且正確的使用指針,下面是初學者常犯的錯誤:
空指針: 指針值為NULL的指針叫空指針,不能運行解引用,一旦解引用空指針就會產(chǎn)生段錯誤。
NULL在大多數(shù)系統(tǒng)的值為0,該地址儲存操作系統(tǒng)重啟的數(shù)據(jù)。
NULL也被當作錯誤標志,如果函數(shù)的返回值是指針類型,當它的值是NULL時說明執(zhí)行出現(xiàn)錯誤。
如何避免空指針產(chǎn)生的段錯誤:對來歷不明的指針進行解引用前要先判斷是否為空
野指針: 指針變量的值是不確定的,隨機的,未知的,這種指針被稱為野指針。
對野指針進行解引用的后果:一切正常 (運氣好)、段錯誤 (大概率)、臟數(shù)據(jù) (堆內(nèi)存申請的越多,臟數(shù)據(jù)可能性越大)。
終結(jié)出來還是那句話:**別人寫的goto我能看懂,但是我自己不會去寫goto!
**
五、編程規(guī)范
我在這里給大家分享一寫我們公司的編程規(guī)范,大家可以學習一下!
- 不允許在C文件中使用“extern”來申明外部函數(shù)或全局變量;
- 禁止使用八進制數(shù);
- bit位變量移植性差,應避免使用。推薦使用boolean類型;
- bit fields位域變量移植性差,不應使用;
- uint, sint使用機器字長,雖然速度快,但有溢出風險,應避免使用;
- 指針的數(shù)學運算只能用在指向數(shù)組或數(shù)組元素的指針上;
- 指針減法只能用在指向同一數(shù)組中元素的指針上;
- 數(shù)組的索引應當是指針數(shù)學運算的唯一可允許的方式;
- 不應在指針類型和整型之間進行強制轉(zhuǎn)換;
- 不應在某類型對象指針和其他不同類型對象指針之間進行強制轉(zhuǎn)換;
- 如果指針所指向的類型帶有const 或volatile 限定符,那么移除限定符的強制轉(zhuǎn)換是不允許的;
- 數(shù)學運算時,應有效防止數(shù)據(jù)溢出;
六、結(jié)語
關(guān)于編程規(guī)范的問題其實還有很多需要注意的事情,如果大家感興趣的話,可以搜索一下網(wǎng)上總結(jié)好的編程規(guī)范范文,尤其是初學者,在最開始就要養(yǎng)成一個良好的編程習慣,不理解的東西就盡量不要使用!
好了,以上就是今天分享的所有內(nèi)容,如有錯誤,歡迎指正。
最后,愿讀到這篇文章的程序員們寫的代碼永無bug!
??
評論