日韩黑丝制服一区视频播放|日韩欧美人妻丝袜视频在线观看|九九影院一级蜜桃|亚洲中文在线导航|青草草视频在线观看|婷婷五月色伊人网站|日本一区二区在线|国产AV一二三四区毛片|正在播放久草视频|亚洲色图精品一区

分享

[zz]scanf()函數(shù)釋疑

 SpringEmpire 2007-09-15
 [zz]scanf()函數(shù)釋疑   文章指數(shù):0  CSDN Blog推出文章指數(shù)概念,文章指數(shù)是對(duì)Blog文章綜合評(píng)分后推算出的,綜合評(píng)分項(xiàng)分別是該文章的點(diǎn)擊量,回復(fù)次數(shù),被網(wǎng)摘收錄數(shù)量,文章長(zhǎng)度和文章類(lèi)型;滿分100,每月更新一次。
scanf()函數(shù)釋疑(上/下)
 
 

作者/收集者:Knocker

                                        scanf()函數(shù)釋疑(上)                               

一、             序言                          

    scanf()函數(shù)是所有C語(yǔ)言學(xué)習(xí)者在學(xué)習(xí)C語(yǔ)言過(guò)程中所遇到的第二個(gè)函數(shù)(第一個(gè)函數(shù)是printf(),Brian W.Kerninghan & Dennis M.Ritchie的“hello,world”程序基本上是所有的C語(yǔ)言學(xué)習(xí)者第一個(gè)范例),所以scanf()函數(shù)應(yīng)當(dāng)是C學(xué)習(xí)者能熟練運(yùn)用的一個(gè)函數(shù),但有很多初學(xué)者對(duì)此函數(shù)不能很好的運(yùn)用,在實(shí)際編程中錯(cuò)誤使用scanf()函數(shù),導(dǎo)至程序產(chǎn)生某種錯(cuò)誤不能正常運(yùn)行,以至產(chǎn)生“scanf()函數(shù)有BUG”,“scanf()函數(shù)無(wú)用論”等等錯(cuò)誤觀點(diǎn)。
    本文結(jié)合筆者在編程實(shí)踐中及論壇上網(wǎng)友所遇到的問(wèn)題作一釋疑,但筆者水平有限(菜鳥(niǎo)級(jí)),難免有謬誤之處,還望達(dá)人指點(diǎn)一二。(Email:knocker.k@126.com)
    本文分上,下兩篇講述了C語(yǔ)言中的scanf()函數(shù)的用法,重點(diǎn)闡述使用scanf()函數(shù)過(guò)程中出現(xiàn)的常見(jiàn)錯(cuò)誤及對(duì)策。當(dāng)然,文中某些解決方法,均可以采用其他函數(shù)和方法來(lái)更好地解決,但本文僅限討論scanf()函數(shù)本身。
    上篇,詳細(xì)介紹了scanf()函數(shù)控制串的構(gòu)成。下篇,用實(shí)際例程介紹scanf()函數(shù)控制串運(yùn)用出現(xiàn)的常見(jiàn)錯(cuò)誤及對(duì)策技巧。

二、      scanf()函數(shù)的控制串

函數(shù)名: scanf
功  能: 執(zhí)行格式化輸入
用  法: int scanf(char *format[,argument,...]);

scanf()函數(shù)是通用終端格式化輸入函數(shù),它從標(biāo)準(zhǔn)輸入設(shè)備(鍵盤(pán)) 讀取輸入的信息。可以讀入任何固有類(lèi)型的數(shù)據(jù)并自動(dòng)把數(shù)值變換成適當(dāng)?shù)臋C(jī)內(nèi)格式。

其調(diào)用格式為:      scanf("<格式化字符串>",<地址表>);

scanf()函數(shù)返回成功賦值的數(shù)據(jù)項(xiàng)數(shù),出錯(cuò)時(shí)則返回EOF。

其控制串由三類(lèi)字符構(gòu)成:

1。格式化說(shuō)明符;
2。空白符;
3。非空白符;

(A)                格式化說(shuō)明符

格式字符           說(shuō)明

%a                 讀入一個(gè)浮點(diǎn)值(僅C99有效)
%A                 同上
%c                 讀入一個(gè)字符
%d                 讀入十進(jìn)制整數(shù)
%i                 讀入十進(jìn)制,八進(jìn)制,十六進(jìn)制整數(shù)
%o                 讀入八進(jìn)制整數(shù)
%x                 讀入十六進(jìn)制整數(shù)
%X                 同上
%c                 讀入一個(gè)字符
%s                 讀入一個(gè)字符串
%f                 讀入一個(gè)浮點(diǎn)數(shù)
%F                 同上
%e                 同上
%E                 同上
%g                 同上
%G                 同上
%p                 讀入一個(gè)指針
%u                 讀入一個(gè)無(wú)符號(hào)十進(jìn)制整數(shù)
%n                 至此已讀入值的等價(jià)字符數(shù)
%[]                掃描字符集合
%%                 讀%符號(hào)
                
附加格式說(shuō)明字符表

修飾符                       說(shuō)明

L/l 長(zhǎng)度修飾符               輸入"長(zhǎng)"數(shù)據(jù)
h 長(zhǎng)度修飾符                 輸入"短"數(shù)據(jù)
W 整型常數(shù)                   指定輸入數(shù)據(jù)所占寬度
* 星號(hào)                       空讀一個(gè)數(shù)據(jù)
hh,ll同上h,l但僅對(duì)C99有效。

(B)         空白字符

空白字符會(huì)使scanf()函數(shù)在讀操作中略去輸入中的一個(gè)或多個(gè)空白字符,空白符可以是space,tab,newline等等,直到第一個(gè)非空白符出現(xiàn)為止。

(C)        非空白字符

一個(gè)非空白字符會(huì)使scanf()函數(shù)在讀入時(shí)剔除掉與這個(gè)非空白字符相同的字符。

注:scanf()控制串知識(shí)就介紹到這里(應(yīng)該比較齊全了^_^),如有遺漏下次補(bǔ)上。下面將結(jié)合實(shí)際例程,一一闡述.

三、      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;
}  

運(yùn)行時(shí)按如下方式輸入三個(gè)值:

3□4□5 ↙(輸入a,b,c的值)

3,4,5 (printf輸出的a,b,c的值)

   (1) &a、&b、&c中的&是地址運(yùn)算符,分別獲得這三個(gè)變量的內(nèi)存地址。
   (2) "%d%d%d"是按十進(jìn)值格式輸入三個(gè)數(shù)值。輸入時(shí),在兩個(gè)數(shù)據(jù)之間可以用一個(gè)或多個(gè)空格、tab鍵、回車(chē)鍵分隔。
      以下是合法輸入方式:
      ① 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;
}  

運(yùn)行時(shí)按如下方式輸入三個(gè)值:

3,4,5 ↙(輸入a,b,c的值)

或者

3,□4,□5 ↙(輸入a,b,c的值)

3,□□□4,□5 ↙(輸入a,b,c的值)
......
都是合法的,但是","一定要跟在數(shù)字后面,如:
3□,4,□5 ↙就非法了,程序出錯(cuò)。(解決方法與原因后面講)

再如:

1、sacnf()中的變量必須使用地址。

         int a, b;
scanf("%d%d",a,b); //錯(cuò)誤
scanf("%d%d",&a,&b);

2、scanf()的格式控制串可以使用其它非空白字符,但在輸入時(shí)必須輸入這些字符。

例:
scanf("%d,%d",&a,&b);
輸入: 3,4 ↙(逗號(hào)與"%d,%d"中的逗號(hào)對(duì)應(yīng))
scanf("a=%d,b=%d",&a,&b);
輸入: a=3,b=4 ↙("a=","b=",逗號(hào)與"%d,%d"中的"a=","b="及逗號(hào)對(duì)應(yīng))

3、在用"%c"輸入時(shí),空格和“轉(zhuǎn)義字符”均作為有效字符。

例:
scanf("%c%c%c",&c1,&c2,&c3);
輸入:a□b□c↙
結(jié)果:a→c1,□→c2,b→c3 (其余被丟棄)

scanf()函數(shù)接收輸入數(shù)據(jù)時(shí),遇以下情況結(jié)束一個(gè)數(shù)據(jù)的輸入:(不是結(jié)束該scanf函數(shù),scanf函數(shù)僅在每一個(gè)數(shù)據(jù)域均有數(shù)據(jù),并按回車(chē)后結(jié)束)。
        ① 遇空格、“回車(chē)”、“跳格”鍵。
        ② 遇寬度結(jié)束。
        ③ 遇非法輸入。

    上篇就寫(xiě)到這里吧,第三小節(jié)的例程"抄"自網(wǎng)上的一個(gè)教程(原因有二:一,可以少打不少字。二,□↙我不知道怎么打。^_^),并刪去其中的錯(cuò)誤之處.這里也順便提醒本文讀者一句:凡事要親力而為,即使是經(jīng)典的書(shū)籍也不免有疏漏之處,所以,用編譯器說(shuō)話是最可靠的,是對(duì)是錯(cuò)請(qǐng)編譯器告訴你。

 

             scanf()函數(shù)釋疑(下)

    在上篇我已經(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ò)示例程序:

#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]);
}

    這實(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)看看這樣行不行:

#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]);
}

    這樣我們就把多維浮點(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!

#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ù)時(shí),遇以下情況結(jié)束一個(gè)數(shù)據(jù)的輸入:(不是結(jié)束該scanf函數(shù),scanf函數(shù)僅在每一個(gè)數(shù)據(jù)域均有數(shù)據(jù),并按回車(chē)后結(jié)束)。
        ① 遇空格、“回車(chē)”、“跳格”鍵。
        ② 遇寬度結(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)證一下:

#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秒,告訴你程序運(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)看下面的程序:

#i nclude "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)題

#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表示出來(lái),啟用printf("c=%d\n",c);這一句,看看scanf()函數(shù)賦給C到底是什么,結(jié)果是 c=10 ,ASCII值為10是什么?換行即\n.對(duì)了,我們每擊打一下"Enter"鍵,向鍵盤(pán)緩沖區(qū)發(fā)去一個(gè)“回車(chē)”(\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);

#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');
}      

這里再給一個(gè)用“空格符”來(lái)處理緩沖區(qū)殘余信息的示例:

運(yùn)行出錯(cuò)的程序:

#i nclude <stdio.h>
int main()
{     int i;
    char j;
    for(i = 0;i < 10;i++)
    {
        scanf("%c",&j);/*這里%前沒(méi)有空格*/
    }
}

使用了空格控制符后:

#i nclude <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ò)?

#i nclude <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ò)誤的類(lèi)型,你的程序不是死鎖,就是得到一個(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)題。

正確的例程:

#i nclude <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



Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=1785678


    本站是提供個(gè)人知識(shí)管理的網(wǎng)絡(luò)存儲(chǔ)空間,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點(diǎn)。請(qǐng)注意甄別內(nèi)容中的聯(lián)系方式、誘導(dǎo)購(gòu)買(mǎi)等信息,謹(jǐn)防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請(qǐng)點(diǎn)擊一鍵舉報(bào)。
    轉(zhuǎn)藏 分享 獻(xiàn)花(0

    0條評(píng)論

    發(fā)表

    請(qǐng)遵守用戶 評(píng)論公約

    類(lèi)似文章 更多