C語言——使用scanf函數時需要注意的問題

轉自:http://blog.csdn.NET/wayne92  

scanf函數我曾經在這個函數上犯過不少錯誤,也看到別人犯過的錯誤,記下來,提醒自己不要重蹈覆轍了。如果對你有用,那就更好了:)如果你發現文章中有錯誤,歡迎你不吝賜 教。希望和大家一起學習!

曾經錯的幾個地方:(xpsp2,vc6.0環境下)

1.空白符問題

[cpp] view plain copy
  1. #include<stdio.h>  
  2.   
  3. void main()  
  4. {     
  5.     int a;  
  6.     printf("input the data\n");  
  7.     scanf("%d\n",&a);// 這裏多了一個回車符\n  
  8.     printf("%d",a);  
  9.     return 0;  
  10. }  
結果要輸入兩個數程序才結束,而不是預期的一個。why?

原因:用空白符結束格式化掃描條件("%d\n")時,scanf會跳過空白符去讀下一個字符,所以你必須再輸入一個數。這裏的空白符包括空格,製表符,換行符,回車符和換頁符。所以如果你用scanf("%d  ",&a)(以空格結束)也會出現同樣的問題。

解決方法:這種錯誤大多是輸入的時候不小心,多注意一點就好了。這種問題也不好檢查,編譯沒有問題,一個空格也不容易看出來。當你的程序出現上面的問題時,自己對照檢查一下就可以了。

2.緩衝區問題

這是一個非常容易錯的地方,我就錯過多次。

[cpp] view plain copy
  1. #include <stdio.h>  
  2.   
  3. void main()  
  4. {  
  5.     char c[5]={0};  
  6.   
  7.     printf("please input 4 characters\n");  
  8.   
  9.     for(int i = 0; i < 4; i++)  
  10.         scanf("%c",&c[i]);  
  11.   
  12.     printf(c);  
  13. }  
如果輸入:
a
b
c

那麼循環就會“提前”結束了.

原因: 輸入a和第一個回車後,a和這個回車符都留在緩衝區中。第一個scanf讀取了a,但是輸入緩衝區裏面還留有一個‘\n’,第二個scanf讀取這個‘\n’。然 後輸入b和第二個回車,同樣的,第三個scanf讀取了b,第四個scanf讀取了第二個回車符‘\n’。第五個讀取了c。所以五個scanf都執行了,並沒有提前結束。只不過有的scanf讀取到了回車符而已。

解決方法:把程序改成這樣就可以了:

[cpp] view plain copy
  1. #include <stdio.h>  
  2.   
  3. void main()  
  4. {  
  5.     char c[5]={0};  
  6.   
  7.     printf("please input 4 characters\n");  
  8.   
  9.     for(int i = 0; i < 4; i++)  
  10.     {  
  11.         scanf("%c",&c[i]);  
  12.         fflush(stdin);  
  13.     }  
  14.     printf(c);  
  15. }  
或者不用scanf,而用gets()函數,如:
[cpp] view plain copy
  1. #include <stdio.h>  
  2.   
  3. void main()  
  4. {  
  5.     char c[5]={0};  
  6.   
  7.     printf("please input 4 characters\n");  
  8.   
  9.     gets(c);  
  10.     printf(c);  
  11. }  
但要注意: 這個函數自動把你最後敲的回車轉換爲字符'\0'。如果你的輸入超過了數組的大小,那麼就會產生錯誤。

3.scanf()函數的參數輸入類型不匹配問題

這是我在csdn論壇上見到的問題,這個錯誤有時候會讓人莫名其妙。

[cpp] view plain copy
  1. #include<stdio.h>  
  2.   
  3. void main()  
  4. {  
  5.     int a=123;  
  6.     char c='a';  
  7.   
  8.     printf("input a int and character\n");  
  9.   
  10.     scanf("%d%c",&a,&c);  
  11.     scanf("%d%c",&a,&c);  
  12.   
  13.     printf("a=%d\nc='%c'\n",a,c);  
  14. }  
當輸入a 回車 後,會直接跳過下面的scanf()語句,直接輸出爲
input a int and character
a
a=123
c='a'
原因: 對於scanf("%d%c",&a,&c),scanf語句執行時,首先試圖從緩衝區中讀入一個%d類型的數據

如果和第一個參數類型匹 配,則繼續從緩衝區中讀取數據和第二個參數進行匹配,依次進行下去,直到匹配完所有的參數;

如果其中有一個參數不匹配,那就從這個地方跳出,忽略這個 scanf後面所有的參數,而去執行下一條語句。

解決方法:scanf() 函數執行成功時的返回值是成功讀取的變量數。也就是說,你這個scanf()函數有幾個變量,如果scanf()函數全部正常讀取,它就返回幾。

但這裏還要注意另一個問題,如果輸入了非法數據,鍵盤緩衝區就可能還個有殘餘信息問題

4. 輸入字符串中有空格問題(很有用)

首先需要介紹scanf()函數中一種很少見但很有用的轉換字符:[...]和[ ^...]

"%[...]" 表示只提取與[]中相匹配的內容,碰到其他內容則終止提取;

"%[^...]"(注意異或符'^')表示提取內容,碰到[]中相匹配的內容則終止提取; 

注意:這裏的匹配是按字符進行的!

[cpp] view plain copy
  1. #include<stdio.h>  
  2.   
  3. void main()   
  4. {   
  5.     char strings[128];   
  6.     scanf("%[1234567890]",strings);   
  7.     printf("%s",strings);  
  8. }   
運行,輸入:1234werew後,結果是:1234

通過運行可以發現它的作用是:如果輸入的字符屬於方括號內字符串中某個字符,那麼就提取該字符;如果一經發現不屬於就結束提取。該方法會自動加上一個字符串結束符到已經提取的字符後面。

scanf("%[^1234567890]",strings); 它的作用是:如果一經發現輸入的字符屬於方括號內字符串中某個字符,那麼就結束提取;如果不屬於就提取該字符。該方法會自動加上一個字符串結束符到已經提取的字符後面。

注意:方括號兩邊不能空格,否則空格也會算在裏面的。

用途:用這種方法可以解決scanf()的輸入中不能有空格的問題。只要用scanf("%[^\n]",strings); 就可以了。

下面是一個實例:

[cpp] view plain copy
  1. #include<stdio.h>  
  2.   
  3. void main()   
  4. {   
  5.     char name[16]={0};  
  6.   
  7.     printf("please input your name\n");  
  8.     scanf("%[^\n]", name);  
  9.   
  10.     printf("your name is %s\n", name);  
  11. }   

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章