scanf函數(shù)
我曾經(jīng)在這個(gè)函數(shù)上犯過不少錯(cuò)誤,也看到別人犯過的錯(cuò)誤,記下來,提醒自己不要重蹈覆轍了。如果對(duì)你有用,那就更好了:)如果你發(fā)現(xiàn)文章中有錯(cuò)誤,歡迎你不吝賜教。希望和大家一起學(xué)習(xí)! 有關(guān)詳細(xì)的scanf函數(shù)解釋,大家可以去看看《C程序設(shè)計(jì)語言》(K&C)和《C語言大全》(后面我把其中scanf的部分貼了出來)。 曾經(jīng)錯(cuò)的幾個(gè)地方:(xpsp2,vc6.0環(huán)境下) 1.空白符問題 #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ì)跳過空白符去讀下一個(gè)字符,所以你必須再輸入一個(gè)數(shù)。這里的空白符包括空格,制表符,換行符,回車符和換頁符。所以如果你用scanf("%d ",&a)也會(huì)出現(xiàn)同樣的問題。 解決方法:這種錯(cuò)誤大多是輸入的時(shí)候不小心,多注意一點(diǎn)就好了。這種問題也不好檢查,編譯沒有問題,一個(gè)空格也不容易看出來。當(dāng)你的程序出現(xiàn)上面的問題時(shí),自己對(duì)照檢查一下就可以了。 2.緩沖區(qū)問題 這是一個(gè)非常容易錯(cuò)的地方,我就錯(cuò)過多次。 #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í)行了,并沒有提前結(jié)束。只不過有的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'。如果你的輸入超過了數(shù)組的大小,那么就會(huì)產(chǎn)生錯(cuò)誤。 3.scanf()函數(shù)的參數(shù)輸入類型不匹配問題 這是我在csdn論壇上見到的問題,這個(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ì)直接跳過下面2個(gè)scanf語句,直接輸出為 123 t 原因:對(duì)于scanf("%d%c",&a,&c),scanf語句執(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à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ù),也就是說,你這個(gè)scanf()函數(shù)有幾個(gè)變量,如果scanf()函數(shù)全部正常讀取,它就返回幾。但這里還要注意另一個(gè)問題,如果輸入了非法數(shù)據(jù),鍵盤緩沖區(qū)就可能還個(gè)有殘余信息問題。 比如: #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中一種很少見但很有用的轉(zhuǎn)換字符:[...]和[ ^...]。 #include<stdio.h> main() { char strings[100]; scanf("%[1234567890]",strings); printf("%s",strings); return 0; } 運(yùn)行,輸入:1234werew后,結(jié)果是:1234。 通過運(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的輸入中不能有空格的問題。只要用 scanf("%[^\n]",strings); 就可以了。很神奇吧。 scanf原型:參見《C語言大全》和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 指向的控制串由以下三類字符組成: ● 格式說明符 ● 空白符 ● 非空白符 轉(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 讀無符號(hào)十進(jìn)制整數(shù) [ ] 掃描字符集合 % 讀 % 符號(hào)(百分號(hào)) 例如: %s 表示讀串而 %d 表示讀整數(shù)。格式串的處理順序?yàn)閺淖蟮接?,格式說明符逐一與變?cè)碇械淖冊(cè)ヅ?。為了讀取長(zhǎng)整數(shù),可以將 l(ell) 放在格式說明符的前面;為了讀取短整數(shù),可以將 h 放在格式說明符的前面。這些修飾符可以與 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 修飾符說明類型 wchar_t 的寬字符指針;也可以與 s 格式代碼一起,用 l 修飾符說明寬字符串的指針。l 修飾符也可以用于修飾掃描集,以說明寬字符。 控制串中的空白符使 scanf() 在輸入流中跳過一個(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。 格式命令可以說明最大域?qū)挕?在百分號(hào)(%)與格式碼之間的整數(shù)用于限制從對(duì)應(yīng)域讀入的最大字符數(shù)。例如,希望向 address 讀入不多于 20 個(gè)字符時(shí),可以書寫成如下形式: scanf( "%20s", address ); 如果輸入流的內(nèi)容多于 20 個(gè)字符,則下次 scanf() 從此次停止處開始讀入。 若達(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é)尾、由讀入字符組成的字符串。 用字符 ^ 可以說明補(bǔ)集。把 ^ 字符放為掃描集的第一字符時(shí),構(gòu)成其它字符組成的命令的補(bǔ)集合,指示 scanf() 只接受未說明的其它字符。 對(duì)于許多實(shí)現(xiàn)來說,用連字符可以說明一個(gè)范圍。 例如,以下掃描集使 scanf() 接受字母 A 到 Z: %[A-Z] 重要的是要注意掃描集是區(qū)分大小寫的。因此,希望掃描大、小寫字符時(shí),應(yīng)該分別說明大、小寫字母。 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。它說明相應(yīng)的變?cè)?signed 或 unsigned char 值,或用于 n 時(shí), 相應(yīng)的變?cè)侵赶?long char 型變量的指針。ll 修飾符也可用于 d、i、o、u、x、X 或 n。它說明相應(yīng)的變?cè)?signed 或者 unsigned long long int 值。 j 格式修飾符應(yīng)用于 d、i、o、u、x、X 或 n,說明匹配的變?cè)穷愋?intmax_t 或 uintmax_t。這些類型在 <stdint.h>; 中聲明,并說明最大寬度的整數(shù)。 z 格式修飾符應(yīng)用于 d、i、o、u、x、X 或 n,說明匹配的變?cè)侵赶?size_t 類型對(duì)象的指針。該類型在 <stddef.h>; 中聲明,并說明 sizeof 的結(jié)構(gòu)。 t 格式修飾符應(yīng)用于 d、i、o、u、x、X 或 n,說明匹配的變?cè)侵赶?ptrdiff_t 類型對(duì)象的指針。該類型在 <stddef.h>; 中聲明,并說明兩個(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ǎng)絡(luò) wangchao.net.cn) |
|