scanf使用全面詳解

scanf("%s",&str);在輸入包含空格的字符串時,容易遇到問題
因爲scanf函數遇到 TAB ⋃⋃ 空格 ⋃⋃ \n 的時候視爲輸入結束。
scanf( “%[^\n]”,str);

%[^\n] 的指定“遇到 \n 結束”,在我們輸入空格的時候,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; 
}  

運行時按如下方式輸入三個值:
345 ↙(輸入a,b,c的值)
345 (printf輸出的a,b,c的值)
  (1&a、&b、&c中的&是地址運算符,分別獲得這三個變量的內存地址。 
  (2"%d%d%d"是按十進值格式輸入三個數值。輸入時,在兩個數據之間可以用一個或多個空格、tab鍵、回車鍵分隔。 
   以下是合法輸入方式: 
   ① 3□□4□□□□5↙ 
   ② 345↙ 
   ③ 3(tab鍵)45
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 ↙就非法了,程序出錯。(解決方法與原因後面講)
再如:
1sacnf()中的變量必須使用地址。
    int a, b; 
scanf("%d%d",a,b); //錯誤 
scanf("%d%d",&a,&b);
2scanf()的格式控制串可以使用其它非空白字符,但在輸入時必須輸入這些字符。
例: 
scanf("%d,%d",&a,&b); 
輸入: 34 ↙(逗號與"%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
b
c
那麼循環就會“提前”結束了.
原因:輸入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
t
原因:對於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 回車
結果是:
2
1
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
t
原因:對於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 回車 
結果是: 
2 
1 
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; 
} 
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章