函數(shù)名: scanf
功 能: 執(zhí)行格式化輸入
用 法: int scanf(char *format[,argument,...]);
scanf()函數(shù)是通用終端格式化輸入函數(shù),它從標(biāo)準(zhǔn)輸入設(shè)備(鍵盤(pán)) 讀取輸入的信息。可以讀入任何固有類型的數(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。
其控制串由三類字符構(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.
#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;
}
運(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鍵、回車鍵分隔。
以下是合法輸入方式:
① 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;
}
運(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ù),并按回車后結(jié)束)。
① 遇空格、“回車”、“跳格”鍵。
② 遇寬度結(jié)束。
③ 遇非法輸入。
問(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);
}
scanf函數(shù)探討
1.空白符問(wèn)題
#include <stdio.h>
main()
{
int a;
printf( "input the data/n ");
scanf( "%d/n ",&a);//這里多了一個(gè)回車符/n
printf( "%d ",a);
return 0;
}
結(jié)果要輸入兩個(gè)數(shù)程序才結(jié)束,而不是預(yù)期的一個(gè)。why?
原因:用空白符結(jié)尾時(shí),scanf會(huì)跳過(guò)空白符去讀下一個(gè)字符,所以你必須再輸入一個(gè)數(shù)。這里的空白符包括
空格,制表符,換行符,回車符和換頁(yè)符。所以如果你用scanf( "%d ",&a)也會(huì)出現(xiàn)同樣的問(wèn)題。
解決方法:這種錯(cuò)誤大多是輸入的時(shí)候不小心,多注意一點(diǎn)就好了。這種問(wèn)題也不好檢查,編譯沒(méi)有問(wèn)題,
一個(gè)空格也不容易看出來(lái)。當(dāng)你的程序出現(xiàn)上面的問(wèn)題時(shí),自己對(duì)照檢查一下就可以了。
2.緩沖區(qū)問(wèn)題
這是一個(gè)非常容易錯(cuò)的地方,我就錯(cuò)過(guò)多次。
#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
b
c
那么循環(huán)就會(huì)“提前”結(jié)束了.
原因:輸入a和第一個(gè)回車后,a和這個(gè)回車符都留在緩沖區(qū)中。第一個(gè)scanf讀取了a,但是輸入緩沖區(qū)里面
還留有一個(gè)/n,第二個(gè)scanf讀取這個(gè)/n。然后輸入b和第二個(gè)回車,同樣的,第三個(gè)scanf讀取了b,第四個(gè)
scanf讀取了第二個(gè)回車符。第五個(gè)讀取了c。所以五個(gè)scanf都執(zhí)行了,并沒(méi)有提前結(jié)束。只不過(guò)有的scanf
讀取到了回車符而已。
解決方法:把程序改成這樣就可以了:
for( i = 0; i < n; i++){
scanf( "%c ",&c[i]);
fflush(stdin);//刷新緩沖區(qū)
}
或者不用scanf,而用gets()函數(shù),如:
#include <stdio.h>
main()
{
char c[5];
gets(c);
printf(c);
return 0;
}
但要注意:這個(gè)函數(shù)自動(dòng)把你最后敲的回車轉(zhuǎn)換為字符 '/0 '。如果你的輸入超過(guò)了數(shù)組的大小,那么就會(huì)產(chǎn)
生錯(cuò)誤。
3.scanf()函數(shù)的參數(shù)輸入類型不匹配問(wèn)題
這是我在csdn論壇上見(jiàn)到的問(wèn)題,這個(gè)錯(cuò)誤有時(shí)候會(huì)讓人莫名其妙。
#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;
}
當(dāng)輸入a 回車 后,會(huì)直接跳過(guò)下面2個(gè)scanf語(yǔ)句,直接輸出為
123
t
原因:對(duì)于scanf( "%d%c ",&a,&c),scanf語(yǔ)句執(zhí)行時(shí),首先試圖從緩沖區(qū)中讀入一個(gè)%d類型的數(shù)據(jù),如果和
第一個(gè)參數(shù)匹配,則繼續(xù)從緩沖區(qū)中讀取數(shù)據(jù)和第二個(gè)參數(shù)進(jìn)行匹配,依次進(jìn)行下去,直到匹配完所有的參
數(shù);如果其中有一個(gè)參數(shù)不匹配,那就從這個(gè)地方跳出,忽略這個(gè)scanf后面所有的參數(shù),而去執(zhí)行下一條語(yǔ)
句。
可以用下面的程序驗(yàn)證一下:
#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 回車
結(jié)果是:
2
1
a
解決方法: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>
main()
{
int a=123,b;
while(scanf( "%d%d ",&a,&b)!=2)
fflush(stdin);
printf( "%d/n%d/n ",a,b);
return 0;
}
你可以試一下,如果輸入不是數(shù)字時(shí),會(huì)有什么反應(yīng)。
補(bǔ)充:scanf中一種很少見(jiàn)但很有用的轉(zhuǎn)換字符:[...]和[ ^...]。
#include <stdio.h>
main()
{
char strings[100];
scanf( "%[1234567890] ",strings);
printf( "%s ",strings);
return 0;
}
運(yùn)行,輸入:1234werew后,結(jié)果是:1234。
通過(guò)運(yùn)行可以發(fā)現(xiàn)它的作用是:如果輸入的字符屬于方括號(hào)內(nèi)字符串中某個(gè)字符,那么就提取該字符;如果
一經(jīng)發(fā)現(xiàn)不屬于就結(jié)束提取。該方法會(huì)自動(dòng)加上一個(gè)字符串結(jié)束符到已經(jīng)提取的字符后面。
scanf( "%[^1234567890] ",strings); 它的作用是:如果一經(jīng)發(fā)現(xiàn)輸入的字符屬于方括號(hào)內(nèi)字符串中某個(gè)字符
,那么就結(jié)束提??;如果不屬于就提取該字符。該方法會(huì)自動(dòng)加上一個(gè)字符串結(jié)束符到已經(jīng)提取的字符后面
。
注意:方括號(hào)兩邊不能空格,如:scanf( "%[ 1234567890 ] ",strings); scanf( "%[ ^1234567890
] ",strings); 不讓空格也會(huì)算在里面的。
用這種方法還可以解決scanf的輸入中不能有空格的問(wèn)題。只要用
scanf( "%[^/n] ",strings); 就可以了。很神奇吧。
scanf原型:參見(jiàn)《C語(yǔ)言大全》和K&C
# include <stdio.h> ;
int scanf( const char *format, ... );
函數(shù) scanf() 是從標(biāo)準(zhǔn)輸入流 stdin 中讀內(nèi)容的通用子程序,可以讀入全部固有類型的數(shù)據(jù)并自動(dòng)轉(zhuǎn)換成機(jī)內(nèi)形式。
在 C99 中,format 用 restrict 修飾。
format 指向的控制串由以下三類字符組成:
● 格式說(shuō)明符
● 空白符
● 非空白符
轉(zhuǎn)換字符(就是%后跟的部分)
a 讀浮點(diǎn)值(僅適用于 C99)
A 讀浮點(diǎn)值(僅適用于 C99)
c 讀單字符
d 讀十進(jìn)制整數(shù)
i 讀十進(jìn)制、八進(jìn)制、十六進(jìn)制整數(shù)
e 讀浮點(diǎn)數(shù)
E 讀浮點(diǎn)數(shù)
f 讀浮點(diǎn)數(shù)
F 讀浮點(diǎn)數(shù)(僅適用于 C99)
g 讀浮點(diǎn)數(shù)
G 讀浮點(diǎn)數(shù)
o 讀八進(jìn)制數(shù)
s 讀字符串
x 讀十六進(jìn)制數(shù)
X 讀十六進(jìn)制數(shù)
p 讀指針值
n 至此已讀入值的等價(jià)字符數(shù)
u 讀無(wú)符號(hào)十進(jìn)制整數(shù)
[ ] 掃描字符集合
% 讀 % 符號(hào)(百分號(hào))
例如: %s 表示讀串而 %d 表示讀整數(shù)。格式串的處理順序?yàn)閺淖蟮接遥袷秸f(shuō)明符逐一與變?cè)碇械?
變?cè)ヅ?。為了讀取長(zhǎng)整數(shù),可以將 l(ell) 放在格式說(shuō)明符的前面;為了讀取短整數(shù),可以將 h 放在格式
說(shuō)明符的前面。這些修飾符可以與 d、i、o、u 和 x 格式代碼一起使用。
默認(rèn)情況下,a、f、e 和 g 告訴 scanf() 為 float 分配數(shù)據(jù)。 如果將 l(ell) 放在這些修飾符的前
面,則 scanf() 為 double 分配數(shù)據(jù)。使用 L 就是告訴 scanf(),接收數(shù)據(jù)的變量是 long double 型變量。
如果使用的現(xiàn)代編譯器程序支持 1995 年增加的寬字符特性, 則可以與 c 格式代碼一起,用 l 修飾符
說(shuō)明類型 wchar_t 的寬字符指針;也可以與 s 格式代碼一起,用 l 修飾符說(shuō)明寬字符串的指針。l 修飾符
也可以用于修飾掃描集,以說(shuō)明寬字符。
控制串中的空白符使 scanf() 在輸入流中跳過(guò)一個(gè)或多個(gè)空白行??瞻追梢允强崭?space)、制表符
(tab)和新行符(newline)。 本質(zhì)上,控制串中的空白符使 scanf() 在輸入流中讀,但不保存結(jié)果,直到發(fā)
現(xiàn)非空白字符為止。
非空白符使 scanf() 在流中讀一個(gè)匹配的字符并忽略之。例如, "%d,%d " 使 scanf() 先讀入一個(gè)整數(shù)
,讀入中放棄逗號(hào),然后讀另一個(gè)整數(shù)。如未發(fā)現(xiàn)匹配,scanf() 返回。
scanf() 中用于保存讀入值的變?cè)仨毝际亲兞恐羔槪聪鄳?yīng)變量的地址。
在輸入流中,數(shù)據(jù)項(xiàng)必須由空格、制表符和新行符分割。逗號(hào)和分號(hào)等不是分隔符,比如以下代碼:
scanf( "%d %d ", &r, &c );
將接受輸入 10 20,但遇到 10,20 則失敗。
百分號(hào)(%)與格式符之間的星號(hào)(*)表示讀指定類型的數(shù)據(jù)但不保存。因此,
scanf( "%d %*c %d ", &x, &y );
對(duì) 10/20 的讀入操作中,10 放入變量 x,20 放入 y。
格式命令可以說(shuō)明最大域?qū)挕?在百分號(hào)(%)與格式碼之間的整數(shù)用于限制從對(duì)應(yīng)域讀入的最大字符數(shù)。
例如,希望向 address 讀入不多于 20 個(gè)字符時(shí),可以書(shū)寫(xiě)成如下形式:
scanf( "%20s ", address );
如果輸入流的內(nèi)容多于 20 個(gè)字符,則下次 scanf() 從此次停止處開(kāi)始讀入。 若達(dá)到最大域?qū)捛耙延?
到空白符,則對(duì)該域的讀立即停止;此時(shí),scanf() 跳到下一個(gè)域。
雖然空格、制表符和新行符都用做域分割符號(hào),但讀單字符操作中卻按一般字符處理。例如,對(duì)輸入流
"x y " 調(diào)用:
scanf( "%c%c%c ", &a, &b, &c );
返回后,x 在變量 a 中,空格在變量 b 中,y 在變量 c 中。
注意,控制串中的其它字符,包括空格、制表符和新行符,都用于從輸入流中匹配并放棄字符,被匹配
的字符都放棄。例如,給定輸入流 "10t20 ",調(diào)用:
scanf( "%dt%d ", &x, &y );
將把 10 和 20 分別放到 x 和 y 中,t 被放棄,因?yàn)?t 在控制串中。
ANSI C 標(biāo)準(zhǔn)向 scanf() 增加了一種新特性,稱為掃描集(scanset)。 掃描集定義一個(gè)字符集合,可由
scanf() 讀入其中允許的字符并賦給對(duì)應(yīng)字符數(shù)組。 掃描集合由一對(duì)方括號(hào)中的一串字符定義,左方括號(hào)前
必須綴以百分號(hào)。 例如,以下的掃描集使 scanf() 讀入字符 A、B 和 C:
%[ABC]
使用掃描集時(shí),scanf() 連續(xù)吃進(jìn)集合中的字符并放入對(duì)應(yīng)的字符數(shù)組,直到發(fā)現(xiàn)不在集合中的字符為
止(即掃描集僅讀匹配的字符)。返回時(shí),數(shù)組中放置以 null 結(jié)尾、由讀入字符組成的字符串。
用字符 ^ 可以說(shuō)明補(bǔ)集。把 ^ 字符放為掃描集的第一字符時(shí),構(gòu)成其它字符組成的命令的補(bǔ)集合,指
示 scanf() 只接受未說(shuō)明的其它字符。
對(duì)于許多實(shí)現(xiàn)來(lái)說(shuō),用連字符可以說(shuō)明一個(gè)范圍。 例如,以下掃描集使 scanf() 接受字母 A 到 Z:
%[A-Z]
重要的是要注意掃描集是區(qū)分大小寫(xiě)的。因此,希望掃描大、小寫(xiě)字符時(shí),應(yīng)該分別說(shuō)明大、小寫(xiě)字母
。
scanf() 返回等于成功賦值的域數(shù)的值,但由于星號(hào)修飾符而讀入未賦值的域不計(jì)算在內(nèi)。給第一個(gè)域
賦值前已出錯(cuò)時(shí),返回 EOF。
C99 為 scanf() 增加了幾個(gè)格式修飾符:hh、ll、j、z 和 t。hh 修飾符可用于 d、i、o、u、x、X 或
n。它說(shuō)明相應(yīng)的變?cè)?signed 或 unsigned char 值,或用于 n 時(shí), 相應(yīng)的變?cè)侵赶?long char 型變
量的指針。ll 修飾符也可用于 d、i、o、u、x、X 或 n。它說(shuō)明相應(yīng)的變?cè)?signed 或者 unsigned long
long int 值。
j 格式修飾符應(yīng)用于 d、i、o、u、x、X 或 n,說(shuō)明匹配的變?cè)穷愋?intmax_t 或 uintmax_t。這些
類型在 <stdint.h> ; 中聲明,并說(shuō)明最大寬度的整數(shù)。
z 格式修飾符應(yīng)用于 d、i、o、u、x、X 或 n,說(shuō)明匹配的變?cè)侵赶?size_t 類型對(duì)象的指針。該類
型在 <stddef.h> ; 中聲明,并說(shuō)明 sizeof 的結(jié)構(gòu)。
t 格式修飾符應(yīng)用于 d、i、o、u、x、X 或 n,說(shuō)明匹配的變?cè)侵赶?ptrdiff_t 類型對(duì)象的指針。
該類型在 <stddef.h> ; 中聲明,并說(shuō)明兩個(gè)指針之間的差別。
例子:
# 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;
}你的問(wèn)題在這個(gè)部分:
3.scanf()函數(shù)的參數(shù)輸入類型不匹配問(wèn)題
這是我在csdn論壇上見(jiàn)到的問(wèn)題,這個(gè)錯(cuò)誤有時(shí)候會(huì)讓人莫名其妙。
#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;
}
當(dāng)輸入a 回車 后,會(huì)直接跳過(guò)下面2個(gè)scanf語(yǔ)句,直接輸出為
123
t
原因:對(duì)于scanf( "%d%c ",&a,&c),scanf語(yǔ)句執(zhí)行時(shí),首先試圖從緩沖區(qū)中讀入一個(gè)%d類型的數(shù)據(jù),如果和
第一個(gè)參數(shù)匹配,則繼續(xù)從緩沖區(qū)中讀取數(shù)據(jù)和第二個(gè)參數(shù)進(jìn)行匹配,依次進(jìn)行下去,直到匹配完所有的參
數(shù);如果其中有一個(gè)參數(shù)不匹配,那就從這個(gè)地方跳出,忽略這個(gè)scanf后面所有的參數(shù),而去執(zhí)行下一條語(yǔ)
句。
可以用下面的程序驗(yàn)證一下:
#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 回車
結(jié)果是:
2
1
a
解決方法: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>
main()
{
int a=123,b;
while(scanf( "%d%d ",&a,&b)!=2)
fflush(stdin);
printf( "%d/n%d/n ",a,b);
return 0;
}
|