scanf詳解

函數名: scanf 
功 能: 運行格式化輸入 
用 法: int scanf(char *format[,argument,...]);
scanf()函數是通用終端格式化輸入函數,它從標準輸入設備(鍵盤) 讀取輸入的信息。能夠讀入不論什麼固有類型的數據並自己主動把數值變換成適當的機內格式。
其調用格式爲:      scanf("<格式化字符串>",<地址表>);
scanf()函數返回成功賦值的數據項數,出錯時則返回EOF。
其控制串由三類字符構成:
1。格式化說明符; 
2。空白符; 
3。非空白符;


(A)                格式化說明符


格式字符           說明
%a                 讀入一個浮點值(僅C99有效) 
%A                 同上 
%c                 讀入一個字符 
%d                 讀入十進制整數 
%i                 讀入十進制,八進制,十六進制整數 
%o                 讀入八進制整數 
%x                 讀入十六進制整數 
%X                 同上 
%c                 讀入一個字符 
%s                 讀入一個字符串 
%f                 讀入一個浮點數 
%F                 同上 
%e                 同上 
%E                 同上 
%g                 同上 
%G                 同上 
%p                 讀入一個指針 
%u                 讀入一個無符號十進制整數 
%n                 至此已讀入值的等價字符數 
%[]                掃描字符集合 
%%                 讀%符號 
                
附加格式說明字符表
修飾符                       說明
L/l 長度修飾符               輸入"長"數據 
h 長度修飾符                 輸入"短"數據 
W 整型常數                   指定輸入數據所佔寬度 
* 星號                       空讀一個數據 
hh,ll同上h,l但僅對C99有效。


(B)         空白字符
空白字符會使scanf()函數在讀操作中略去輸入中的一個或多個空白字符,空白符能夠是space,tab,newline等等,直到第一個非空白符出現爲止。
(C)        非空白字符
一個非空白字符會使scanf()函數在讀入時剔除掉與這個非空白字符同樣的字符。


注:scanf()控制串知識就介紹到這裏(應該比較齊全了^_^),如有遺漏下次補上。以下將結合實際例程,一一闡述.
三、      scanf()函數的控制串的使用
例1.
#include "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中的&是地址運算符,分別獲得這三個變量的內存地址。 
   (2) "%d%d%d"是按十進值格式輸入三個數值。輸入時,在兩個數據之間能夠用一個或多個空格、tab鍵、回車鍵分隔。 
      下面是合法輸入方式: 
      ① 3□□4□□□□5↙ 
      ② 3↙ 
         4□5↙ 
      ③ 3(tab鍵)4↙ 
         5↙


例2.
#include "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的值) 
...... 
都是合法的,可是","一定要跟在數字後面,如: 
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"中的逗號相應) 
scanf("a=%d,b=%d",&a,&b); 
輸入: a=3,b=4 ↙("a=","b=",逗號與"%d,%d"中的"a=","b="及逗號相應)


3、在用"%c"輸入時,空格和“轉義字符”均作爲有效字符。
例: 
scanf("%c%c%c",&c1,&c2,&c3); 
輸入:a□b□c↙ 
結果:a→c1,□→c2,b→c3 (其餘被丟棄)


scanf()函數接收輸入數據時,遇下面情況結束一個數據的輸入:(不是結束該scanf函數,scanf函數僅在每個數據域均有數據,並按回車後結束)。 
        ① 遇空格、“回車”、“跳格”鍵。 
        ② 遇寬度結束。 
        ③ 遇非法輸入。
問題二:scanf()函數不能正確接受有空格的字符串?如: I love you!


#include <stdio.h> 
int main() 

    char str[80]; 
    
    scanf("%s",str); 
    printf("%s",str);
    return 0; 
}
輸入:I live you! 
輸出:I
       scanf()函數接收輸入數據時,遇下面情況結束一個數據的輸入:(不是結束該scanf函數,scanf函數僅在每個數據域均有數據,並按回車後結束)。 
        ① 遇空格、“回車”、“跳格”鍵。 
        ② 遇寬度結束。 
        ③ 遇非法輸入。
所以,上述程序並不能達到預期目的,scanf()掃描到"I"後面的空格就覺得對str的賦值結束,並忽略後面的"love you!".這裏要注意是"love you!"還在鍵盤緩衝區(關於這個問題,網上我所見的說法都是如此,可是,我經過調試發現,事實上這時緩衝區字符串首尾指針已經相等了,也就是說緩衝區清空了,scanf()函數應該僅僅是掃描stdin流,這個殘存信息是在stdin中)。我們修改一下上面的程序來驗證一下:
#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秒,告訴你程序執行到什麼地方*/ 
    scanf("%s",str1);/*這兩句無需你再輸入,是對鍵盤盤緩衝區再掃描   */ 
    scanf("%s",str2);/*這兩句無需你再輸入,是對鍵盤盤緩衝區再掃描    */ 
    printf("/n%s",str1); 
    printf("/n%s",str2); 
    return 0; 
}
輸入:I love you! 
輸出:I 
      love 
      you!
好了,原因知道了,那麼scanf()函數能不能完畢這個任務?回答是:能!別忘了scanf()函數另一個 %[] 格式控制符(假設對%[]不瞭解的請查看本文的上篇),請看以下的程序:
#include "stdio.h" 
int main() 

    char string[50]; 
    
     /*scanf("%s",string);不能接收空格符*/ 
     scanf("%[^/n]",string); 
     printf("%s/n",string); 
     return 0; 
}
問題三:鍵盤緩衝區殘餘信息問題


#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表示出來,啓用printf("c=%d/n",c);這一句,看看scanf()函數賦給C究竟是什麼,結果是 c=10 ,ASCII值爲10是什麼?換行即/n.對了,我們每擊打一下"Enter"鍵,向鍵盤緩衝區發去一個“回車”(/r),一個“換行"(/n),在這裏/r被scanf()函數處理掉了(姑且這麼覺得吧^_^),而/n被scanf()函數“錯誤”地賦給了c.
解決的方法:能夠在兩個scanf()函數之後加個fflush(stdin);,還有加getch(); getchar();也能夠,可是要視詳細scanf()語句加那個,這裏就不分析了,讀者自己去摸索吧。可是加fflush(stdin);無論什麼情況都可行。
函數名: fflush 
功 能: 清除一個流 
用 法: 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'); 
}     
這裏再給一個用“空格符”來處理緩衝區殘餘信息的演示樣例:
執行出錯的程序:
#include <stdio.h> 
int main() 

    int i; 
    char j; 
    for(i = 0;i < 10;i++) 
    { 
        scanf("%c",&j);/*這裏%前沒有空格*/ 
    } 
}
使用了空格控制符後:
#include <stdio.h> 
int main() 

    int i; 
    char j; 
    for(i = 0;i < 10;i++) 
    { 
        scanf(" %c",&j);/*注意這裏%前有個空格*/ 
    } 
}
    能夠執行看看兩個程序有什麼不同。
問題四   怎樣處理scanf()函數誤輸入造成程序死鎖或出錯?
#include <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的值,那麼沒什麼問題,可是,你不能保證使用者每一次都能正確輸入,一旦輸入了錯誤的類型,你的程序不是死鎖,就是得到一個錯誤的結果,呵呵,這可能全部人都遇到過的問題吧?
解決方法:scanf()函數運行成功時的返回值是成功讀取的變量數,也就是說,你這個scanf()函數有幾個變量,假設scanf()函數所有正常讀取,它就返回幾。但這裏還要注意還有一個問題,假設輸入了非法數據,鍵盤緩衝區就可能還個有殘餘信息問題。
正確的例程:
#include <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);
}


 


 


 


scanf函數探討 


1.空白符問題 
#include <stdio.h> 
main() 

int a; 
printf( "input the data/n "); 
scanf( "%d/n ",&a);//這裏多了一個回車符/n 
printf( "%d ",a); 
return 0; 

結果要輸入兩個數程序才結束,而不是預期的一個。why? 
原因:用空白符結尾時,scanf會跳過空白符去讀下一個字符,所以你必須再輸入一個數。這裏的空白符包含 


空格,製表符,換行符,回車符和換頁符。所以假設你用scanf( "%d ",&a)也會出現相同的問題。 
解決方法:這樣的錯誤大多是輸入的時候不小心,多注意一點就好了。這樣的問題也不好檢查,編譯沒有問題, 


一個空格也不easy看出來。當你的程序出現上面的問題時,自己對比檢查一下就能夠了。 




2.緩衝區問題 
這是一個很easy錯的地方,我就錯過多次。 
#include <stdio.h> 
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和第一個回車後,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 <stdio.h> 
main() 

char c[5]; 
gets(c); 
printf(c); 
return 0; 

但要注意:這個函數自己主動把你最後敲的回車轉換爲字符 '/0 '。假設你的輸入超過了數組的大小,那麼就會產 


生錯誤。 




3.scanf()函數的參數輸入類型不匹配問題 
這是我在csdn論壇上見到的問題,這個錯誤有時候會讓人莫名其妙。 
#include <stdio.h> 
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 

原因:對於scanf( "%d%c ",&a,&c),scanf語句運行時,首先試圖從緩衝區中讀入一個%d類型的數據,假設和 


第一個參數匹配,則繼續從緩衝區中讀取數據和第二個參數進行匹配,依次進行下去,直到匹配全然部的參 


數;假設當中有一個參數不匹配,那就從這個地方跳出,忽略這個scanf後面全部的參數,而去運行下一條語 


句。 
能夠用以下的程序驗證一下: 
#include <stdio.h> 
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 回車 
結果是: 



解決方法:scanf()函數運行成功時的返回值是成功讀取的變量數,也就是說,你這個scanf()函數有幾個變量 


,假設scanf()函數所有正常讀取,它就返回幾。但這裏還要注意還有一個問題,假設輸入了非法數據,鍵盤緩 


衝區就可能還個有殘餘信息問題。 
比方: 
#include <stdio.h> 
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 <stdio.h> 
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 <stdio.h> ; 
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。這些 


類型在 <stdint.h> ; 中聲明,並說明最大寬度的整數。 
z 格式修飾符應用於 d、i、o、u、x、X 或 n,說明匹配的變元是指向 size_t 類型對象的指針。該類 


型在 <stddef.h> ; 中聲明,並說明 sizeof 的結構。 
t 格式修飾符應用於 d、i、o、u、x、X 或 n,說明匹配的變元是指向 ptrdiff_t 類型對象的指針。 


該類型在 <stddef.h> ; 中聲明,並說明兩個指針之間的區別。 


樣例: 


# include <stdio.h> ; 
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; 
}你的問題在這個部分: 
3.scanf()函數的參數輸入類型不匹配問題 
這是我在csdn論壇上見到的問題,這個錯誤有時候會讓人莫名其妙。 
#include <stdio.h> 
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 

原因:對於scanf( "%d%c ",&a,&c),scanf語句運行時,首先試圖從緩衝區中讀入一個%d類型的數據,假設和 


第一個參數匹配,則繼續從緩衝區中讀取數據和第二個參數進行匹配,依次進行下去,直到匹配全然部的參 


數;假設當中有一個參數不匹配,那就從這個地方跳出,忽略這個scanf後面全部的參數,而去運行下一條語 


句。 
能夠用以下的程序驗證一下: 
#include <stdio.h> 
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 回車 
結果是: 



解決方法:scanf()函數運行成功時的返回值是成功讀取的變量數,也就是說,你這個scanf()函數有幾個變量 


,假設scanf()函數所有正常讀取,它就返回幾。但這裏還要注意還有一個問題,假設輸入了非法數據,鍵盤緩 


衝區就可能還個有殘餘信息問題。 
比方: 
#include <stdio.h> 
main() 

int a=123,b; 
while(scanf( "%d%d ",&a,&b)!=2) 
fflush(stdin); 
printf( "%d/n%d/n ",a,b); 
return 0; 
發佈了83 篇原創文章 · 獲贊 14 · 訪問量 10萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章