1. 程式人生 > >C語言scanf函式用法詳細解釋!

C語言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)也會出現同樣的問題。 
解決方法:這種錯誤大多是輸入的時候不小心,多注意一點就好了。這種問題也不好檢查,編譯沒有問題, 


一個空格也不容易看出來。當你的程式出現上面的問題時,自己對照檢查一下就可以了。 




2.緩衝區問題 
這是一個非常容易錯的地方,我就錯過多次。 
#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;