曾經錯的幾個地方:(xpsp2,vc6.0環境下)
1.空白符問題
#include
main()
{
int a;
printf("input the data\n");
scanf("%d\n",&a);//這里多了一個回車符\n
printf("%d",a);
return 0;
}
結果要輸入兩個數程序才結束,而不是預期的一個。why?
原因:用空白符結尾時,scanf會跳過空白符去讀下一個字符,所以你必須再輸入一個數。這里的空白符包括空格,制表符,換行符,回車符等。所以如果你用scanf("%d ",&a)也會出現同樣的問題。
解決方法:這種錯誤大多是輸入的時候不小心,多注意一點就好了。這種問題也不好檢查,編譯沒有問題,一個空格也不容易看出來。當你的程序出現上面的問題時,自己對照檢查一下就可以了。
2.緩沖區問題(“垃圾”字符)
這是一個非常容易錯的地方,我就錯過多次。
#include
main()
{
int n = 5;
char c[n];
for(int i = 0; i < n; i++)
c[i] = scanf("%c",&c[i]);
printf(c);
return 0;
}
如果輸入:
a
b
c
那么循環就會“提前”結束了.
原因:輸入a和第一個回車后,a和這個回車符都留在緩沖區中。第一個scanf讀取了a,但是輸入緩沖區里面還留有一個\n,第二個scanf讀取這個\n。然后輸入b和第二個回車,同樣的,第三個scanf讀取了b,第四個scanf讀取了第二個回車符。第五個讀取了c。所以五個scanf都執行了,并沒有提前結束。只不過有的scanf讀取到了回車符而已。
解決方法:把程序改成這樣就可以了:
for( i = 0; i < n; i++){
scanf("%c",&c[i]);
fflush(stdin);//刷新緩沖區
}
或者不用scanf,而用gets()函數,如:
#include
main()
{
char c[5];
gets(c);
printf(c);
return 0;
}
但要注意:這個函數自動把你后敲的回車轉換為字符'\0'。如果你的輸入超過了數組的大小,那么就會產生錯誤。
3.scanf()函數的參數輸入類型不匹配問題
這是我在csdn論壇上見到的問題,這個錯誤有時候會讓人莫名其妙。
#include
main()
{
int a=123;
char c='t';
printf("input\n");
scanf("%d%c",&a,&c);
scanf("%d%c",&a,&c);
scanf("%d%c",&a,&c);
printf("%d\n%c\n",a,c);
return 0;
}
當輸入a 回車 后,會直接跳過下面2個scanf語句,直接輸出為
123
t
原因:對于scanf("%d%c",&a,&c),scanf語句執行時,首先試圖從緩沖區中讀入一個%d類型的數據,如果和第一個參數匹配,則繼續從緩沖區中讀取數據和第二個參數進行匹配,依次進行下去,直到匹配完所有的參數;
如果其中有一個參數不匹配,那就從這個地方跳出,忽略這個scanf后面所有的參數,而去執行下一條語句。
可以用下面的程序驗證一下:
#include
int main()
{
int a=123,b=1;
char c='t';
scanf("%d%d",&a,&b);
scanf("%c",&c);
printf("%d\n%d\n%c\n",a,b,c);
return 0;
}輸入:2 回車a 回車
結果是:
2
1
a
解決方法:scanf()函數執行成功時的返回值是成功讀取的變量數,也就是說,你這個scanf()函數有幾個變量,如果scanf()函數全部正常讀取,它就返回幾。但這里還要注意另一個問題,如果輸入了非法數據,鍵盤緩沖區就可能還個有殘余信息問題。
比如:
#include
main()
{
int a=123,b;
while(scanf("%d%d",&a,&b)!=2)
fflush(stdin);
printf("%d\n%d\n",a,b);
return 0;
}
你可以試一下,如果輸入不是數字時,會有什么反應。
補充:scanf中一種很少見但很有用的轉換字符:[...]和[ ^...]。
#include
main()
{
char strings[100];
scanf("%[1234567890]",strings);
printf("%s",strings);
return 0;
}
運行,輸入:1234werew后,結果是:1234。
通過運行可以發現它的作用是:如果輸入的字符屬于方括號內字符串中某個字符,那么就提取該字符;如果一經發現不屬于就結束提取。該方法會自動加上一個字符串結束符到已經提取的字符后面。
scanf("%[^1234567890]",strings); 它的作用是:如果一經發現輸入的字符屬于方括號內字符串中某個字符,那么就結束提取;如果不屬于就提取該字符。該方法會自動加上一個字符串結束符到已經提取的字符后面。
注意:方括號兩邊不能空格,如:scanf("%[ 1234567890 ]",strings); scanf("%[ ^1234567890 ]",strings); 不讓空格也會算在里面的。
用這種方法還可以解決scanf的輸入中不能有空格的問題。只要用
scanf("%[^\n]",strings); 就可以了。很神奇吧。
scanf原型:參見《C語言大全》和K&C
# include
int scanf( const char *format, ... );
函數 scanf() 是從標準輸入流 stdin 中讀內容的通用子程序,可以讀入全部固有類型的數據并自動轉換成機內形式。
在 C99 中,format 用 restrict 修飾。
format 指向的控制串由以下三類字符組成:
● 格式說明符
● 空白符
● 非空白符
轉換字符(就是%后跟的部分)
a 讀浮點值(僅適用于 C99)
A 讀浮點值(僅適用于 C99)
c 讀單字符
d 讀十進制整數
i 讀十進制、八進制、十六進制整數
e 讀浮點數
E 讀浮點數
f 讀浮點數
F 讀浮點數(僅適用于 C99)
g 讀浮點數
G 讀浮點數
o 讀八進制數
s 讀字符串
x 讀十六進制數
X 讀十六進制數
p 讀指針值
n 至此已讀入值的等價字符數
u 讀無符號十進制整數
[ ] 掃描字符集合
% 讀 % 符號(百分號)
例如: %s 表示讀串而 %d 表示讀整數。格式串的處理順序為從左到右,格式說明符逐一與變元表中的變元匹配。為了讀取長整數,可以將 l(ell) 放在格式說明符的前面;為了讀取短整數,可以將 h 放在格式說明符的前面。這些修飾符可以與 d、i、o、u 和 x 格式代碼一起使用。
默認情況下,a、f、e 和 g 告訴 scanf() 為 float 分配數據。 如果將 l(ell) 放在這些修飾符的前面,則 scanf() 為 double 分配數據。使用 L 就是告訴 scanf(),接收數據的變量是 long double 型變量。
如果使用的現代編譯器程序支持 1995 年增加的寬字符特性, 則可以與 c 格式代碼一起,用 l 修飾符說明類型 wchar_t 的寬字符指針;也可以與 s 格式代碼一起,用 l 修飾符說明寬字符串的指針。l 修飾符也可以用于修飾掃描集,以說明寬字符。
控制串中的空白符使 scanf() 在輸入流中跳過一個或多個空白行。空白符可以是空格(space)、制表符(tab)和新行符(newline)。 本質上,控制串中的空白符使 scanf() 在輸入流中讀,但不保存結果,直到發現非空白字符為止。
非空白符使 scanf() 在流中讀一個匹配的字符并忽略之。例如,"%d,%d" 使 scanf() 先讀入一個整數,讀入中放棄逗號,然后讀另一個整數。如未發現匹配,scanf() 返回。
scanf() 中用于保存讀入值的變元必須都是變量指針,即相應變量的地址。
在輸入流中,數據項必須由空格、制表符和新行符分割。逗號和分號等不是分隔符,比如以下代碼:
scanf( "%d %d", &r, &c );
將接受輸入 10 20,但遇到 10,20 則失敗。
百分號(%)與格式符之間的星號(*)表示讀指定類型的數據但不保存。因此,
scanf( "%d %*c %d", &x, &y );
對 10/20 的讀入操作中,10 放入變量 x,20 放入 y。
格式命令可以說明大域寬。 在百分號(%)與格式碼之間的整數用于限制從對應域讀入的大字符數。例如,希望向 address 讀入不多于 20 個字符時,可以書寫成如下形式:
scanf( "%20s", address );
如果輸入流的內容多于 20 個字符,則下次 scanf() 從此次停止處開始讀入。 若達到大域寬前已遇到空白符,則對該域的讀立即停止;此時,scanf() 跳到下一個域。
雖然空格、制表符和新行符都用做域分割符號,但讀單字符操作中卻按一般字符處理。例如,對輸入流 "x y" 調用:
scanf( "%c%c%c", &a, &b, &c );
返回后,x 在變量 a 中,空格在變量 b 中,y 在變量 c 中。
注意,控制串中的其它字符,包括空格、制表符和新行符,都用于從輸入流中匹配并放棄字符,被匹配的字符都放棄。例如,給定輸入流 "10t20",調用:
scanf( "%dt%d", &x, &y );
將把 10 和 20 分別放到 x 和 y 中,t 被放棄,因為 t 在控制串中。
ANSI C 標準向 scanf() 增加了一種新特性,稱為掃描集(scanset)。 掃描集定義一個字符集合,可由 scanf() 讀入其中允許的字符并賦給對應字符數組。 掃描集合由一對方括號中的一串字符定義,左方括號前必須綴以百分號。 例如,以下的掃描集使 scanf() 讀入字符 A、B 和 C:
%[ABC]
使用掃描集時,scanf() 連續吃進集合中的字符并放入對應的字符數組,直到發現不在集合中的字符為止(即掃描集僅讀匹配的字符)。返回時,數組中放置以 null 結尾、由讀入字符組成的字符串。
用字符 ^ 可以說明補集。把 ^ 字符放為掃描集的第一字符時,構成其它字符組成的命令的補集合,指示 scanf() 只接受未說明的其它字符。
對于許多實現來說,用連字符可以說明一個范圍。 例如,以下掃描集使 scanf() 接受字母 A 到 Z:
%[A-Z]
重要的是要注意掃描集是區分大小寫的。因此,希望掃描大、小寫字符時,應該分別說明大、小寫字母。
scanf() 返回等于成功賦值的域數的值,但由于星號修飾符而讀入未賦值的域不計算在內。給第一個域賦值前已出錯時,返回 EOF。
C99 為 scanf() 增加了幾個格式修飾符:hh、ll、j、z 和 t。hh 修飾符可用于 d、i、o、u、x、X 或 n。它說明相應的變元是 signed 或 unsigned char 值,或用于 n 時, 相應的變元是指向 long char 型變量的指針。ll 修飾符也可用于 d、i、o、u、x、X 或 n。它說明相應的變元是 signed 或者 unsigned long long int 值。
j 格式修飾符應用于 d、i、o、u、x、X 或 n,說明匹配的變元是類型 intmax_t 或 uintmax_t。這些類型在
z 格式修飾符應用于 d、i、o、u、x、X 或 n,說明匹配的變元是指向 size_t 類型對象的指針。該類型在
t 格式修飾符應用于 d、i、o、u、x、X 或 n,說明匹配的變元是指向 ptrdiff_t 類型對象的指針。該類型在
例子:
# include
int main( void )
{
char str[80], str2[80];
int i;
/* read a string and a integer */
scanf( "%s%d", str, &i );
/* read up to 79 chars into str */
scanf( "%79s", str );
/* skip the integer between the two strings */
scanf( "%s%*d%s", str, str2 );
return 0;
}