例:
#include <stdio.h>
int main(void)
{
int a = 0;
scanf("%5d",&a);
printf("a = %d",a);
return 0;
}
我運行這個代碼,在輸入整數小於等於5位的時候沒問題,輸出正確。當輸入整數多於5位的時候,它只會讀取到第五位,而不讀取剩下的輸入。那麼那些多於五位的輸入是丟棄了嗎?不是的,這又是scanf的一個特性。當scanf()未能讀取指定形式的輸入時,它就留下這個不相容的輸入,以供下次進行讀取。
修改上述代碼
#include <stdio.h>
int main(void)
{
int a = 0;
int b = 0;
scanf("%5d",&a);
printf("a = %d\n",a);
scanf("%d",&b);
printf("b = %d\n",b);
return 0;
}
運行結果如下
這個特性如果不加注意,可能在後續編程中遇到某些問題調試不出來,所以還是有必要理解一下的。
再看一個例子
#include <stdio.h>
int main(void)
{
int a;
int status = 0;
printf("Enter an integer:(q to quit)");
status = scanf("%d",&a);
while(status = 1)
{
printf("Enter next integer:(q to quit)");
status = scanf("%d",&a);
}
printf("Finished");
return 0;
}
這個例子本意是為了告誡大家,不要在應該使用==的地方使用=。
但也可以說明scanf的上述特性。運行該程序會發現,不輸入q時程序運行正常,在輸入q後,程序開始無限循環,用戶沒有機會進行更多的輸入。WHY?
這是因為,輸入q,status被設置為0,但是在循環判斷時又把status重置為1並開始另一次循環。BESIDES,scanf想讀取的使%d類型,即整型,而q不是整型,所以q不被讀取,留在緩沖區等待下一個scanf讀取。而當到下次循環時,讀取前面留下來的q,scanf再次失敗,這就是為什麼用戶無法再輸入的原因。
scanf("%d",&a);該語句要讀入一個整數,並將其賦給a。
當讀入的不是整數時,有下列情況
1. 輸入的是空白字符(制表符,空格,換行符),scanf會跳過空白字符直到第一個非空白字符處,然後它會一直讀取字符,直到遇到空白字符,或遇到一個不符合正在讀取的類型的字符。所以scanf("%d%d",&a,&b); scanf("%d %d",&a,&b); scanf("%d %d",&a,&b);的行為是相同的
2. 輸入的是不符合正在讀取類型的字符,比如字母,它會保留這個不相容的字符,留待下一個scanf讀入。
浮點數也類似。
但是對於%c來說,情況略微有所不同。
因為對於%c說明符,所有的輸入字符都是平等的。如果下一個輸入字符是一個空格或者換行符(不論是本次輸入的還是之前遺留下來的),將會把這個空格或換行符賦給指定的變量;不會跳過空白字符。
但是!向格式字符串中添加一個空格將會導致一些區別。
命令scanf("%c",&ch);讀取在輸入中遇到的第一個字符,而scanf(" %c",&ch);則讀取遇到的第一個非空白字符。
注意空格的位置是在%c整體之前,而非% c
這為我們提供了一種吃掉緩沖區換行符的思路。以往如果在scanf("%c",&ch);之前又scanf,我們不得不用fflush(stdin);或者添加一個%*c來吃掉換行符。但是現在,我們有了新方法,而且更加簡便易行。那就是在以scanf(" %c",&ch);取代scanf("%c",&ch);
格式字符串中的空格意味著跳過下一個輸入項之前的任何空格,注意,任何空格的概念也包括沒有空格的特殊情況。
scanf("%d, %d",&a,&b);將會接受下列任何一行輸入
88,21
88, 21
88, 21 注意逗號是英文標點,且逗號要緊跟第一個數字
scanf("%d , %d",&a,&b);將會接受下列任何一行輸入
88,21
88 , 21
88, 21
88 ,
21
———————————————————————————————————————————————————————————————————————————