典型錯誤1:指針指向
上述代碼意圖比較明顯:定義了一個int變量a和指針變量pa,并且把a的地址給了指針pa。接著通過鍵入給a賦值,但運行結果如下:
其實這個問題是我們學習指針的時候的一個典型錯誤了,我們知道調用scanf函數給變量賦值時,賦值對象要為地址的形式,通常是加取址符“&”,但是這里采用的是*pa的格式,這里涉及的指針相關知識前面給大家講過,為了更好地理解本題,就再重復一下: 對于指針來說,有己址、己值、它址、它值等特點,己址就是指針變量本身的地址,己值就是指針變量本身地址所存放的值,也就是我們通常說的指向的地址,這也正是它址,所以己值和它址意義是一樣的,而它值就是指針指向地址位置所存放的值。 而這里的*pa表示的意義就是它值a,那就是說這么寫的話下面兩行代碼是等價的:
scanf("%d", *pa);scanf("%d", a);
對比過后顯然是錯誤的,大家一眼看出a要寫成&a,這沒問題。但也有人說可以把*pa改成&pa,這樣行嗎?其實這么說的人還是對指針中己值和己址的概念沒搞清楚,&pa表示的意義是己址,即指針變量本身的地址,就是說你試圖用scanf修改指針變量本身地址上的值,而這個值原本是變量a的地址,其實就是在修改指針的指向!正確的寫法應該這樣:
scanf("%d",pa);
pa表示a的地址,即為它址,也就是&a,所以上面寫法才與下面的等價:
scanf("%d", &a);
典型錯誤2:getchar函數
char c;while((c=getchar())!=EOF){...}
這段代碼的本意是用getchar函數讀取緩沖區字符直到結束,但是在編譯運行時,發現上面幾行代碼一直報錯!邏輯上沒問題啊,那這究竟錯在哪里?讀者可以自己思考一下再往下看。
其實產生報錯的原因有兩點,一個是對getchar函數理解不到位,另一個是EOF的問題。
我們首先來說說getchar函數的問題,標準庫中給出了該函數的使用說明:在它讀取一個字符后,會將其轉換為int類型返回,所以首先char c要改為int c,關于getchar的問題還沒講完,后面還要說。 我們接著來看看EOF的問題,初學者對它的理解經常會有偏差,首先它是一個宏,定義于頭文件,為-1;其次它并不是很多人理解的文件結束符,實際上它是一個標志位,區別于其他所有字符的存在,表示一種沒有其他字符的信號。 講到這里,我們再回到getchar函數,由上面可以看出它的返回值必須是一個能包含所有字符的數據類型,方便它表示任意字符和EOF等標志位。 因此,上面代碼的錯誤就很明顯了,可能有兩種情況:1.如果編譯器中的char是有符號的且EOF被定義為-1,而恰好有字符等于0xff,那么getchar就會提前結束。當然,如果輸入全部是7位以下的字符,那很長時間不會有錯誤。2.如果編譯器中的char是無符號的,則實際的EOF值會被截斷,不再會識別為EOF,將會陷入無限循環。
這里肯定會有人問我們鍵入-1來模擬EOF跳出循環不行嗎?實際上是不行的,-1是有-和1兩個字符組成的,而getchar一次只能讀取一個字符,所以上述代碼EOF與從鍵盤輸入的字符無關,那這豈不是只能死循環了?當然不是,我們可以通過按鍵組合ctrl+d或者ctrl+z來指示結束,當然,這里的按鍵組合輸入只是我們的一種約定,不應該顯示檢查按鍵組合的值。
典型錯誤3:存儲機制
char *p = NULL;p = "hello world";strcpy(p, "hello world");
題目很簡單,就問這段代碼寫的有沒有問題,如果有,問題在哪里?
其實這個問題如果你對C語言的存儲機制非常熟悉的話,應該是很簡單的:我們簡單分析一下,第一行代碼是沒問題的,第二行意思是讓指針p指向字符串常量,單看也沒問題,而問題就出在第三行了,它的意圖是對指針p指向地址的內容進行修改,當然還用“hello world”只是為了增加點迷惑性而已。
上面說到了C語言的內存機制,其實第二行代碼過后,hello world 作為字符串常量存放于內存中的常量區,且是只讀,而此時指針p存放的是字符串常量的地址,第三行代碼企圖通過strcpy修改只讀段的內容,因此很明顯會報錯,這也是這三行代碼的問題所在了。
關于C的存儲問題,可能有的人還不太了解,那就借這個機會簡單給大家提幾句,這也是以前我寫過的問題:
一個編譯的C程序占用的內存分為以下幾個部分:
1、棧區(stack)—也稱自動類型存儲區,由編譯器自動分配釋放,存放函數的參數值,局部變量的值等,例如函數調用結束后自動釋放。
2、堆區(heap)—也稱動態分配內存區,由程序員分配釋放,從分配到程序結束為止,若不釋放,程序結束時可能由OS回收,比如malloc分配的內存,free釋放的內存。
3、全局區(靜態區)(static)—全局變量和靜態變量的存儲是放在一塊的,初始化的全局變量和靜態變量在一塊區域,未初始化的全局變量和未初始化的靜態變量在相鄰的另一塊區域,程序結束后由系統釋放。
4、文字常量區—常量字符串放在這里,程序結束后由系統釋放。
5、程序代碼區—編譯后的程序代碼放在這里。 來看一個具體的C程序
怎么樣?問題雖然簡單,但也給我們以后寫代碼提了個醒,這種不易察覺的錯誤大家一定要小心再小心,盡量避免,就說到這里吧,感謝大家耐心閱讀!
-END-
-
C語言
+關注
關注
180文章
7629瀏覽量
140120 -
函數
+關注
關注
3文章
4367瀏覽量
64055
原文標題:3個C語言編程易犯的錯誤:也許你也犯過(附代碼)
文章出處:【微信號:gh_c472c2199c88,微信公眾號:嵌入式微處理器】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
評論