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

評(píng)論