在上篇我已經(jīng)表達(dá)了兩個(gè)觀點(diǎn),這里再重申一次:1。本文僅對(duì)scanf()函數(shù)控制串運(yùn)用進(jìn)行探討,本文所有例程并不構(gòu)成編程建議。2。凡事要親力而為,不同平臺(tái)不同編譯器,可能會(huì)有不同結(jié)果。本文所有例程均在WIN-TC+windows Me下調(diào)試。
四、scanf()函數(shù)控制串運(yùn)用出現(xiàn)的常見(jiàn)錯(cuò)誤及對(duì)策技巧
問(wèn)題一: 程序編譯通過(guò),但運(yùn)行錯(cuò)誤提示如下:
scanf : floating point formats not linked
Abnormal program termination
出錯(cuò)示例程序:
#include <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]);
}
這實(shí)際上是個(gè)與本文主題無(wú)關(guān)的問(wèn)題,也是與scanf()函數(shù)無(wú)關(guān),是編譯器的問(wèn)題。
原因很明確:沒(méi)有鏈接浮點(diǎn)庫(kù)。早期系統(tǒng)內(nèi)存資源緊張,多維浮點(diǎn)數(shù)組占用內(nèi)存量大(一維浮點(diǎn)數(shù)組就沒(méi)有此問(wèn)題),因此TC在編譯時(shí)盡量不加入無(wú)關(guān)的部分,在沒(méi)發(fā)現(xiàn)需要浮點(diǎn)轉(zhuǎn)換程序時(shí),就不在可執(zhí)行程序中安裝這個(gè)部分。而有時(shí)TC又不能正確識(shí)別實(shí)際上確實(shí)需要做浮點(diǎn)轉(zhuǎn)換,因此就會(huì)出現(xiàn)上面錯(cuò)誤。
解決的方法:告訴TC需要做浮點(diǎn)數(shù)的輸入轉(zhuǎn)換。將以下語(yǔ)句加入上面程序中標(biāo)有/*這里*/處。
方法一: float c;
scanf("%f",&c);
方法二: float c,*t;//此處手誤,現(xiàn)已更正&t===》*t;
t=&c;
.....
也就是說(shuō),編譯器只要有浮點(diǎn)轉(zhuǎn)換的線索,TC就會(huì)把浮點(diǎn)轉(zhuǎn)換連上,所以一般大一點(diǎn)的程序里的就會(huì)有浮點(diǎn)變量反而沒(méi)有此問(wèn)題。
但問(wèn)題到此并沒(méi)結(jié)束,我在上面有句“一維浮點(diǎn)數(shù)組就沒(méi)有此問(wèn)題”,那么我們來(lái)看看這樣行不行:
#include <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]);
}
這樣我們就把多維浮點(diǎn)數(shù)組降為一維浮點(diǎn)數(shù)組來(lái)處理,調(diào)試一下,程序運(yùn)行正常。這說(shuō)明TC編譯器僅在處理多維浮點(diǎn)數(shù)組(結(jié)構(gòu)體)有此“未鏈接浮點(diǎn)庫(kù)”的問(wèn)題。
問(wèn)題二:scanf()函數(shù)不能正確接受有空格的字符串?如: I love you!
#include <stdio.h>
int main()
{
char str[80];
scanf("%s",str);
printf("%s",str);
return 0;
}
輸入:I live you!
輸出:I
scanf()函數(shù)接收輸入數(shù)據(jù)時(shí),遇以下情況結(jié)束一個(gè)數(shù)據(jù)的輸入:(不是結(jié)束該scanf函數(shù),scanf函數(shù)僅在每一個(gè)數(shù)據(jù)域均有數(shù)據(jù),并按回車后結(jié)束)。
① 遇空格、“回車”、“跳格”鍵。
② 遇寬度結(jié)束。
③ 遇非法輸入。
所以,上述程序并不能達(dá)到預(yù)期目的,scanf()掃描到"I"后面的空格就認(rèn)為對(duì)str的賦值結(jié)束,并忽略后面的"love you!".這里要注意是"love you!"還在鍵盤(pán)緩沖區(qū)(關(guān)于這個(gè)問(wèn)題,網(wǎng)上我所見(jiàn)的說(shuō)法都是如此,但是,我經(jīng)過(guò)調(diào)試發(fā)現(xiàn),其實(shí)這時(shí)緩沖區(qū)字符串首尾指針已經(jīng)相等了,也就是說(shuō)緩沖區(qū)清空了,scanf()函數(shù)應(yīng)該只是掃描stdin流,這個(gè)殘存信息是在stdin中)。我們改動(dòng)一下上面的程序來(lái)驗(yàn)證一下:
#include <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秒,告訴你程序運(yùn)行到什么地方*/
scanf("%s",str1);/*這兩句無(wú)需你再輸入,是對(duì)鍵盤(pán)盤(pán)緩沖區(qū)再掃描 */
scanf("%s",str2);/*這兩句無(wú)需你再輸入,是對(duì)鍵盤(pán)盤(pán)緩沖區(qū)再掃描 */
printf("\n%s",str1);
printf("\n%s",str2);
return 0;
}
輸入:I love you!
輸出:I
love
you!
好了,原因知道了,那么scanf()函數(shù)能不能完成這個(gè)任務(wù)?回答是:能!別忘了scanf()函數(shù)還有一個(gè) %[] 格式控制符(如果對(duì)%[]不了解的請(qǐng)查看本文的上篇),請(qǐng)看下面的程序:
#include "stdio.h"
int main()
{
char string[50];
/*scanf("%s",string);不能接收空格符*/
scanf("%[^\n]",string);
printf("%s\n",string);
return 0;
}
問(wèn)題三:鍵盤(pán)緩沖區(qū)殘余信息問(wèn)題
#include <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表示出來(lái),啟用printf("c=%d\n",c);這一句,看看scanf()函數(shù)賦給C到底是什么,結(jié)果是 c=10 ,ASCII值為10是什么?換行即\n.對(duì)了,我們每擊打一下"Enter"鍵,向鍵盤(pán)緩沖區(qū)發(fā)去一個(gè)“回車”(\r),一個(gè)“換行"(\n),在這里\r被scanf()函數(shù)處理掉了(姑且這么認(rèn)為吧^_^),而\n被scanf()函數(shù)“錯(cuò)誤”地賦給了c.
解決辦法:可以在兩個(gè)scanf()函數(shù)之后加個(gè)fflush(stdin);,還有加getch(); getchar();也可以,但是要視具體scanf()語(yǔ)句加那個(gè),這里就不分析了,讀者自己去摸索吧。但是加fflush(stdin);不管什么情況都可行。
函數(shù)名: fflush
功 能: 清除一個(gè)流
用 法: int fflush(FILE *stream);
#include <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');
}
這里再給一個(gè)用“空格符”來(lái)處理緩沖區(qū)殘余信息的示例:
運(yùn)行出錯(cuò)的程序:
#include <stdio.h>
int main()
{
int i;
char j;
for(i = 0;i < 10;i++)
{
scanf("%c",&j);/*這里%前沒(méi)有空格*/
}
}
使用了空格控制符后:
#include <stdio.h>
int main()
{
int i;
char j;
for(i = 0;i < 10;i++)
{
scanf(" %c",&j);/*注意這里%前有個(gè)空格*/
}
}
可以運(yùn)行看看兩個(gè)程序有什么不同。
問(wèn)題四 如何處理scanf()函數(shù)誤輸入造成程序死鎖或出錯(cuò)?
#include <stdio.h>
int main()
{
int a,b,c; /*計(jì)算a+b*/
scanf("%d,%d",&a,&b);
c=a+b;
printf("%d+%d=%d",a,b,c);
}
如上程序,如果正確輸入a,b的值,那么沒(méi)什么問(wèn)題,但是,你不能保證使用者每一次都能正確輸入,一旦輸入了錯(cuò)誤的類型,你的程序不是死鎖,就是得到一個(gè)錯(cuò)誤的結(jié)果,呵呵,這可能所有人都遇到過(guò)的問(wèn)題吧?
解決方法:scanf()函數(shù)執(zhí)行成功時(shí)的返回值是成功讀取的變量數(shù),也就是說(shuō),你這個(gè)scanf()函數(shù)有幾個(gè)變量,如果scanf()函數(shù)全部正常讀取,它就返回幾。但這里還要注意另一個(gè)問(wèn)題,如果輸入了非法數(shù)據(jù),鍵盤(pán)緩沖區(qū)就可能還個(gè)有殘余信息問(wèn)題。
正確的例程:
#include <stdio.h>
int main()
{
int a,b,c; /*計(jì)算a+b*/
while(scanf("%d,%d",&a,&b)!=2)fflush(stdin);
c=a+b;
printf("%d+%d=%d",a,b,c);
}
就此結(jié)束此文吧,最后還得照例謙虛幾句,本人水平有限(的的確確有限^_^,這到是真話),謬誤難免還望達(dá)人指點(diǎn)一二,在下在此謝過(guò)了.
(全文完)
knocker 2004.10.21