作者/收集者:Knocker
scanf()函數(shù)釋疑(上)
一、 序言
scanf()函數(shù)是所有C語言學(xué)習(xí)者在學(xué)習(xí)C語言過程中所遇到的第二個函數(shù)(第一個函數(shù)是printf(),Brian W.Kerninghan & Dennis M.Ritchie的“hello,world”程序基本上是所有的C語言學(xué)習(xí)者第一個范例),所以scanf()函數(shù)應(yīng)當(dāng)是C學(xué)習(xí)者能熟練運用的一個函數(shù),但有很多初學(xué)者對此函數(shù)不能很好的運用,在實際編程中錯誤使用scanf()函數(shù),導(dǎo)至程序產(chǎn)生某種錯誤不能正常運行,以至產(chǎn)生“scanf()函數(shù)有BUG”,“scanf()函數(shù)無用論”等等錯誤觀點。
本文結(jié)合筆者在編程實踐中及論壇上網(wǎng)友所遇到的問題作一釋疑,但筆者水平有限(菜鳥級),難免有謬誤之處,還望達(dá)人指點一二。(Email:knocker.k@126.com)
本文分上,下兩篇講述了C語言中的scanf()函數(shù)的用法,重點闡述使用scanf()函數(shù)過程中出現(xiàn)的常見錯誤及對策。當(dāng)然,文中某些解決方法,均可以采用其他函數(shù)和方法來更好地解決,但本文僅限討論scanf()函數(shù)本身。
上篇,詳細(xì)介紹了scanf()函數(shù)控制串的構(gòu)成。下篇,用實際例程介紹scanf()函數(shù)控制串運用出現(xiàn)的常見錯誤及對策技巧。
二、 scanf()函數(shù)的控制串
函數(shù)名: scanf
功 能: 執(zhí)行格式化輸入
用 法: int scanf(char *format[,argument,...]);
scanf()函數(shù)是通用終端格式化輸入函數(shù),它從標(biāo)準(zhǔn)輸入設(shè)備(鍵盤) 讀取輸入的信息??梢宰x入任何固有類型的數(shù)據(jù)并自動把數(shù)值變換成適當(dāng)?shù)臋C內(nèi)格式。
其調(diào)用格式為: scanf("<格式化字符串>",<地址表>);
scanf()函數(shù)返回成功賦值的數(shù)據(jù)項數(shù),出錯時則返回EOF。
其控制串由三類字符構(gòu)成:
1。格式化說明符;
2??瞻追?;
3。非空白符;
(A) 格式化說明符
格式字符 說明
%a 讀入一個浮點值(僅C99有效)
%A 同上
%c 讀入一個字符
%d 讀入十進制整數(shù)
%i 讀入十進制,八進制,十六進制整數(shù)
%o 讀入八進制整數(shù)
%x 讀入十六進制整數(shù)
%X 同上
%c 讀入一個字符
%s 讀入一個字符串
%f 讀入一個浮點數(shù)
%F 同上
%e 同上
%E 同上
%g 同上
%G 同上
%p 讀入一個指針
%u 讀入一個無符號十進制整數(shù)
%n 至此已讀入值的等價字符數(shù)
%[] 掃描字符集合
%% 讀%符號
附加格式說明字符表
修飾符 說明
L/l 長度修飾符 輸入"長"數(shù)據(jù)
h 長度修飾符 輸入"短"數(shù)據(jù)
W 整型常數(shù) 指定輸入數(shù)據(jù)所占寬度
* 星號 空讀一個數(shù)據(jù)
hh,ll同上h,l但僅對C99有效。
(B) 空白字符
空白字符會使scanf()函數(shù)在讀操作中略去輸入中的一個或多個空白字符,空白符可以是space,tab,newline等等,直到第一個非空白符出現(xiàn)為止。
(C) 非空白字符
一個非空白字符會使scanf()函數(shù)在讀入時剔除掉與這個非空白字符相同的字符。
注:scanf()控制串知識就介紹到這里(應(yīng)該比較齊全了^_^),如有遺漏下次補上。下面將結(jié)合實際例程,一一闡述.
三、 scanf()函數(shù)的控制串的使用
例1.
#i nclude "stdio.h"
int main(void)
{
int a,b,c;
scanf("%d%d%d",&a,&b,&c);
printf("%d,%d,%d\n",a,b,c);
return 0;
}
運行時按如下方式輸入三個值:
3□4□5 ↙(輸入a,b,c的值)
3,4,5 (printf輸出的a,b,c的值)
(1) &a、&b、&c中的&是地址運算符,分別獲得這三個變量的內(nèi)存地址。
(2) "%d%d%d"是按十進值格式輸入三個數(shù)值。輸入時,在兩個數(shù)據(jù)之間可以用一個或多個空格、tab鍵、回車鍵分隔。
以下是合法輸入方式:
① 3□□4□□□□5↙
② 3↙
4□5↙
③ 3(tab鍵)4↙
5↙
例2.
#i nclude "stdio.h"
int main(void)
{
int a,b,c;
scanf("%d,%d,%d",&a,&b,&c);
printf("%d,%d,%d\n",a,b,c);
return 0;
}
運行時按如下方式輸入三個值:
3,4,5 ↙(輸入a,b,c的值)
或者
3,□4,□5 ↙(輸入a,b,c的值)
3,□□□4,□5 ↙(輸入a,b,c的值)
......
都是合法的,但是","一定要跟在數(shù)字后面,如:
3□,4,□5 ↙就非法了,程序出錯。(解決方法與原因后面講)
再如:
1、sacnf()中的變量必須使用地址。
int a, b;
scanf("%d%d",a,b); //錯誤
scanf("%d%d",&a,&b);
2、scanf()的格式控制串可以使用其它非空白字符,但在輸入時必須輸入這些字符。
例:
scanf("%d,%d",&a,&b);
輸入: 3,4 ↙(逗號與"%d,%d"中的逗號對應(yīng))
scanf("a=%d,b=%d",&a,&b);
輸入: a=3,b=4 ↙("a=","b=",逗號與"%d,%d"中的"a=","b="及逗號對應(yīng))
3、在用"%c"輸入時,空格和“轉(zhuǎn)義字符”均作為有效字符。
例:
scanf("%c%c%c",&c1,&c2,&c3);
輸入:a□b□c↙
結(jié)果:a→c1,□→c2,b→c3 (其余被丟棄)
scanf()函數(shù)接收輸入數(shù)據(jù)時,遇以下情況結(jié)束一個數(shù)據(jù)的輸入:(不是結(jié)束該scanf函數(shù),scanf函數(shù)僅在每一個數(shù)據(jù)域均有數(shù)據(jù),并按回車后結(jié)束)。
① 遇空格、“回車”、“跳格”鍵。
② 遇寬度結(jié)束。
③ 遇非法輸入。
上篇就寫到這里吧,第三小節(jié)的例程"抄"自網(wǎng)上的一個教程(原因有二:一,可以少打不少字。二,□↙我不知道怎么打。^_^),并刪去其中的錯誤之處.這里也順便提醒本文讀者一句:凡事要親力而為,即使是經(jīng)典的書籍也不免有疏漏之處,所以,用編譯器說話是最可靠的,是對是錯請編譯器告訴你。
scanf()函數(shù)釋疑(下)
在上篇我已經(jīng)表達(dá)了兩個觀點,這里再重申一次:1。本文僅對scanf()函數(shù)控制串運用進行探討,本文所有例程并不構(gòu)成編程建議。2。凡事要親力而為,不同平臺不同編譯器,可能會有不同結(jié)果。本文所有例程均在WIN-TC+windows Me下調(diào)試。
四、 scanf()函數(shù)控制串運用出現(xiàn)的常見錯誤及對策技巧
問題一: 程序編譯通過,但運行錯誤提示如下:
scanf : floating point formats not linked
Abnormal program termination
出錯示例程序:
#i nclude <stdio.h>
int main(void)
{ int i,j ;
float s[3][3];
/*這里*/
for(i=0;i<3;i++)
for(j=0;j<3;j++)
scanf("%f",&s[i][j]);
for(i=0;i<3;i++)
for(j=0;j<3;j++)
printf("%f",s[i][j]);
}
這實際上是個與本文主題無關(guān)的問題,也是與scanf()函數(shù)無關(guān),是編譯器的問題。
原因很明確:沒有鏈接浮點庫。早期系統(tǒng)內(nèi)存資源緊張,多維浮點數(shù)組占用內(nèi)存量大(一維浮點數(shù)組就沒有此問題),因此TC在編譯時盡量不加入無關(guān)的部分,在沒發(fā)現(xiàn)需要浮點轉(zhuǎn)換程序時,就不在可執(zhí)行程序中安裝這個部分。而有時TC又不能正確識別實際上確實需要做浮點轉(zhuǎn)換,因此就會出現(xiàn)上面錯誤。
解決的方法:告訴TC需要做浮點數(shù)的輸入轉(zhuǎn)換。將以下語句加入上面程序中標(biāo)有/*這里*/處。
方法一: float c;
scanf("%f",&c);
方法二: float c,*t;//此處手誤,現(xiàn)已更正&t===》*t;
t=&c;
.....
也就是說,編譯器只要有浮點轉(zhuǎn)換的線索,TC就會把浮點轉(zhuǎn)換連上,所以一般大一點的程序里的就會有浮點變量反而沒有此問題。
但問題到此并沒結(jié)束,我在上面有句“一維浮點數(shù)組就沒有此問題”,那么我們來看看這樣行不行:
#i nclude <stdio.h>
int main(void)
{ int i,j ;
float s[3][3],*ptr;
ptr=&s[0][0];
for(i=0;i<3;i++)
for(j=0;j<3;j++)
scanf("%f",ptr+i*3+j);
for(i=0;i<3;i++)
for(j=0;j<3;j++)
printf("%7.2f\n",s[i][j]);
}
這樣我們就把多維浮點數(shù)組降為一維浮點數(shù)組來處理,調(diào)試一下,程序運行正常。這說明TC編譯器僅在處理多維浮點數(shù)組(結(jié)構(gòu)體)有此“未鏈接浮點庫”的問題。
問題二:scanf()函數(shù)不能正確接受有空格的字符串?如: I love you!
#i nclude <stdio.h>
int main()
{ char str[80];
scanf("%s",str);
printf("%s",str);
return 0;
}
輸入:I live you!
輸出:I
scanf()函數(shù)接收輸入數(shù)據(jù)時,遇以下情況結(jié)束一個數(shù)據(jù)的輸入:(不是結(jié)束該scanf函數(shù),scanf函數(shù)僅在每一個數(shù)據(jù)域均有數(shù)據(jù),并按回車后結(jié)束)。
① 遇空格、“回車”、“跳格”鍵。
② 遇寬度結(jié)束。
③ 遇非法輸入。
所以,上述程序并不能達(dá)到預(yù)期目的,scanf()掃描到"I"后面的空格就認(rèn)為對str的賦值結(jié)束,并忽略后面的"love you!".這里要注意是"love you!"還在鍵盤緩沖區(qū)(關(guān)于這個問題,網(wǎng)上我所見的說法都是如此,但是,我經(jīng)過調(diào)試發(fā)現(xiàn),其實這時緩沖區(qū)字符串首尾指針已經(jīng)相等了,也就是說緩沖區(qū)清空了,scanf()函數(shù)應(yīng)該只是掃描stdin流,這個殘存信息是在stdin中)。我們改動一下上面的程序來驗證一下:
#i nclude <stdio.h>
int main()
{ char str[80];
char str1[80];
char str2[80];
scanf("%s",str);/*此處輸入:I love you! */
printf("%s",str);
sleep(5);/*這里等待5秒,告訴你程序運行到什么地方*/
scanf("%s",str1);/*這兩句無需你再輸入,是對鍵盤盤緩沖區(qū)再掃描 */
scanf("%s",str2);/*這兩句無需你再輸入,是對鍵盤盤緩沖區(qū)再掃描 */
printf("\n%s",str1);
printf("\n%s",str2);
return 0;
}
輸入:I love you!
輸出:I
love
you!
好了,原因知道了,那么scanf()函數(shù)能不能完成這個任務(wù)?回答是:能!別忘了scanf()函數(shù)還有一個 %[] 格式控制符(如果對%[]不了解的請查看本文的上篇),請看下面的程序:
#i nclude "stdio.h"
int main()
{ char string[50];
/*scanf("%s",string);不能接收空格符*/
scanf("%[^\n]",string);
printf("%s\n",string);
return 0;
}
問題三:鍵盤緩沖區(qū)殘余信息問題
#i nclude <stdio.h>
int main()
{ int a;
char c;
do
{
scanf("%d",&a);
scanf("%c",&c);
printf("a=%d c=%c\n",a,c);
/*printf("c=%d\n",c);*/
}while(c!='N');
}
scanf("%c",&c);這句不能正常接收字符,什么原因呢?我們用printf("c=%d\n",c);將C用int表示出來,啟用printf("c=%d\n",c);這一句,看看scanf()函數(shù)賦給C到底是什么,結(jié)果是 c=10 ,ASCII值為10是什么?換行即\n.對了,我們每擊打一下"Enter"鍵,向鍵盤緩沖區(qū)發(fā)去一個“回車”(\r),一個“換行"(\n),在這里\r被scanf()函數(shù)處理掉了(姑且這么認(rèn)為吧^_^),而\n被scanf()函數(shù)“錯誤”地賦給了c.
解決辦法:可以在兩個scanf()函數(shù)之后加個fflush(stdin);,還有加getch(); getchar();也可以,但是要視具體scanf()語句加那個,這里就不分析了,讀者自己去摸索吧。但是加fflush(stdin);不管什么情況都可行。
函數(shù)名: fflush
功 能: 清除一個流
用 法: int fflush(FILE *stream);
#i nclude <stdio.h>
int main()
{ int a;
char c;
do
{
scanf("%d",&a);
fflush(stdin);
scanf("%c",&c);
fflush(stdin);
printf("a=%d c=%c\n",a,c);
}while(c!='N');
}
這里再給一個用“空格符”來處理緩沖區(qū)殘余信息的示例:
運行出錯的程序:
#i nclude <stdio.h>
int main()
{ int i;
char j;
for(i = 0;i < 10;i++)
{
scanf("%c",&j);/*這里%前沒有空格*/
}
}
使用了空格控制符后:
#i nclude <stdio.h>
int main()
{ int i;
char j;
for(i = 0;i < 10;i++)
{
scanf(" %c",&j);/*注意這里%前有個空格*/
}
}
可以運行看看兩個程序有什么不同。
問題四 如何處理scanf()函數(shù)誤輸入造成程序死鎖或出錯?
#i nclude <stdio.h>
int main()
{ int a,b,c; /*計算a+b*/
scanf("%d,%d",&a,&b);
c=a+b;
printf("%d+%d=%d",a,b,c);
}
如上程序,如果正確輸入a,b的值,那么沒什么問題,但是,你不能保證使用者每一次都能正確輸入,一旦輸入了錯誤的類型,你的程序不是死鎖,就是得到一個錯誤的結(jié)果,呵呵,這可能所有人都遇到過的問題吧?
解決方法:scanf()函數(shù)執(zhí)行成功時的返回值是成功讀取的變量數(shù),也就是說,你這個scanf()函數(shù)有幾個變量,如果scanf()函數(shù)全部正常讀取,它就返回幾。但這里還要注意另一個問題,如果輸入了非法數(shù)據(jù),鍵盤緩沖區(qū)就可能還個有殘余信息問題。
正確的例程:
#i nclude <stdio.h>
int main()
{ int a,b,c; /*計算a+b*/
while(scanf("%d,%d",&a,&b)!=2)fflush(stdin);
c=a+b;
printf("%d+%d=%d",a,b,c);
}
就此結(jié)束此文吧,最后還得照例謙虛幾句,本人水平有限(的的確確有限^_^,這到是真話),謬誤難免還望達(dá)人指點一二,在下在此謝過了.
(全文完)
knocker 2004.10.21