1. 程式人生 > >C語言——使用scanf函式時需要注意的問題

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

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

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

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

1.空白符問題

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

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

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

2.緩衝區問題

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

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

那麼迴圈就會“提前”結束了.

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

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

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

3.scanf()函式的引數輸入型別不匹配問題

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

  1. #include<stdio.h>
  2. void main()  
  3. {  
  4.     int a=123;  
  5.     char c='a';  
  6.     printf("input a int and character\n");  
  7.     scanf("%d%c",&a,&c);  
  8.     scanf("%d%c",&a,&c);  
  9.     printf("a=%d\nc='%c'\n",a,c);  
  10. }  
當輸入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()函式中一種很少見但很有用的轉換字元:[...]和[ ^...]

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

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

注意:這裡的匹配是按字元進行的!

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

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

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

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

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

下面是一個例項:

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