scanf()函数是格式化输入函数,它从标准输入设备(键盘) 读取输入的信息。其调用格式为: scanf("<格式化字符串>",<地址表>);scanf()函数返回成功赋值的数据项数,出错时则返回EOF。 格式化字符串包括以下三类不同的字符; 1、 格式化说明符:
格式字符 说明 %a 读入一个浮点值(仅C99有效) %A 同上 %c 读入一个字符 %d 读入十进制整数 %i 读入十进制,八进制,十六进制整数 %o 读入八进制整数 %x 读入十六进制整数 %X 同上 %c 读入一个字符 %s 读入一个 %f 读入一个浮点数 %F 同上 %e 同上 %E 同上 %g 同上 %G 同上 %p 读入一个指针 %u 读入一个无符号十进制整数 %n 至此已读入值的等价字符数 %[] 扫描字符集合 %% 读%符号 %* 指定类型的数据但不保存 比如: 百分号(%)与格式符之间的星号(*)表示读指定类型的数据但不保存。因此, int i, j; printf("i, j=?\n"); //scanf("%d, %d", &i, &j); //scanf("%d, %*c, %d", &i, &j); scanf( "%d %*d %d", &i, &j); 在输入10 20 30 的操作中,10 放入变量i,30 放入j。忽略了20。 附加格式说明字符表 修饰符 说明 L/l 长度修饰符 输入"长"数据 h 长度修饰符 输入"短"数据 W 整型常数 指定输入数据所占宽度 * 星号 空读一个数据 hh,ll同上h,l但仅对C99标准才有效。 2、空白字符 空白字符会使scanf()函数在读操作中略去输入中的一个或多个空白字符,空白符可以是space,tab,newline等等,直到第一个非空白符出现为止。
3、非空白字符 一个非空白字符会使scanf()函数在读入时剔除掉与这个非空白字符相同的字符。但在输入时必须输入这些字符。否则就会出错 。
#include <stdio.h>
int main() { /*char str[25]; int k, n; while (scanf("%s%d", str, &k) != EOF) { n = strlen(str); if (k > n) k %= n; RightReverse(str, k, n); } */ int i, j; printf("i, j=?\n"); scanf("%d, %d", &i, &j); }
上述例子中的scanf()函数先读一个整型数,然后把接着输入的逗号剔除掉,最后读入另一个整型数。如果","这一特定字符没有找到,scanf()函数就终止。注明一下,这必须是小写下的“,”不能是大写下的。否则也将视为找不到。读者自己可以尝试一下。
同时也应该注意,在输入的时候是允许输入空格的。如下面的例子:
int a,b,c; scanf("%d,%d,%d",&a,&b,&c); printf("%d,%d,%d/n",a,b,c);
运行时按如下方式输入三个值:
1,2,3 ↙(输入a,b,c的值)
或者
1,□2,□3 ↙(输入a,b,c的值)(注明□表示空格)
1,□□□2,□3 ↙(输入a,b,c的值)
......
都是合法的,但是","一定要跟在数字后面,如:
1□,2,□3 ↙就非法了,程序出错。
注明:当输入“,”的时候,一定要注意是不是英文输入法下的,否则也会出错。
若scanf参数之间的分隔符不是“,”的时候,则参数之间必须输入一个或多个空格或者Tab。
当使用如下输入方式时:
int i, j; printf("i, j=?\n"); //scanf("%d, %d", &i, &j); //scanf("%d, %*c, %d", &i, &j); scanf( "%d%d", &i, &j);//scanf( "%d %d", &i, &j);也可行
//输入时,是通过一个或者多个空格也可以使用Tab键来进行参数的隔离。均都将输入的数字分别赋值给i和j。此时,不能使用“,来进行参数的隔离。
例:
scanf("%d%d%d",&a,&b,&c); printf("%d,%d,%d%\n",a,b,c);//
运行时按如下方式输入三个值:
1□2□3 ↙(输入a,b,c的值)(□表示空格)
1,2,3 (printf输出的a,b,c的值,注意有输出“,”。
(1) &a、&b、&c中的&是地址运算符,分别获得这三个变量的内存地址。
(2) "%d%d%d"是按十进值格式输入三个数值。输入时,在两个数据之间可以用一个或多个空格、tab键、回车键分隔。
以下是合法输入方式:
① 1□□2□□□□3↙
② 1↙
2□3↙
③ 1(tab键)2↙
3↙
所以,Enter的使用在参数未使用完之前都是无妨碍的。
此前所说的scanf()的格式控制串可以使用其它非空白字符,但在输入时必须输入这些字符。就是已经包含在上面的例子中了。即在scanf("%d,%d",&a,&b); 中在%d之后出现的“,”则在输入的时候,也必须要输入。
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="及逗号对应,在输入的过程中必须输入的,否则会出错)
补充说明:
1、在用"%c"输入时,空格和“转义字符”均作为有效字符。如下使用:
char a,b,c;
scanf("%c%c%c",&a,&b,&c);
输入:a□b□c↙
结果:a→a,□→b,b→c (其余被丢弃)
此外:补充说明对于输入是字符串的情况。
#include <iostream> #include<string.h> //using namespace std; int main() { char str[80]; //std::string s1="I am here"; //std::string str;//="I am here"; scanf("%s",str); printf("%s",str); }
2、若是在输入的时候,输入了 I can do 包含有空格的字符序列的话,则遇到空格会停止。具体来说 scanf()函数接收输入数据时,遇以下情况结束一个数据的输入:(不是结束该scanf函数,scanf函数仅在每一个数据域均有数据,并按回车后结束)。
① 遇空格、“回车”、tab键。
② 遇宽度结束。
③ 遇非法输入。
但是空格之后的内容仍旧在键盘缓冲区。
可以采用以下的代码来验证。
#include <stdio.h> #include <Windows.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); system("pause"); return 0; }
但是别忘了,scanf()函数还有一个 %[] 格式控制符,可以通过该控制符来实现。
#include <stdio.h> //#include <Windows.h> int main() { char str[80]; char string[50]; scanf("%[^\n]",string); printf("%s\n",string); return 0; }
3、键盘缓冲区残余信息问题
先看下以下的例子:
#include <stdio.h> //#include <Windows.h> int main() { int a; char c; /*do {*/ scanf("%d",&a); //fflush(stdin); //c=getchar(); scanf("%c",&c); printf("c=%d\n",c);//ASCII码 printf("a=%d c=%c\n",a,c);//结果是 c=10 ,ASCII值为10是什么?换行即/n. //每击打一下"Enter"键,向键盘缓冲区发去一个“回车”(/r),一个“换行"(/n), //在这里/r被scanf()函数处理掉了(姑且这么认为吧^_^),而/n被scanf()函数“错误”地赋给了c. /*printf("c=%d/n",c);*/ //}while(c!='N'); //解决办法:可以在两个scanf()函数之后加个fflush(stdin);, //还有加getch(); getchar(); 也可以,但是要视具体scanf()语句加那个,这里就不分析了,读者自己去摸索吧。 //但是加fflush(stdin);不管什么情况都可行。 }
函数 fflush 的作用是 清除一个流 。用 法: int fflush(FILE *stream);
#include <stdio.h> //#include <Windows.h> int main() { int a; char c; do { scanf("%d",&a); fflush(stdin);//fflush用于清空缓冲流,虽然一般感觉不到,但是默认printf是缓冲输出的。 scanf("%c",&c);//在没有fflush(stdin)时候, scanf("%c",&c)这句不能正常接收字符,因为a接受完毕之后需要按下enter键, //进行将‘\n’传给了c。 fflush(stdin); printf("a=%d c=%c\n",a,c); }while(c!='N'); }
#include <stdio.h> //#include <Windows.h> int main() { int i; char j; for(i = 0;i < 10;i++) { scanf("%c",&j);/*这里%前没有空格*/ //每一次循环,都是先将输入的字符给j,然后再将enter的‘\n’给j. printf("j=%c\n",j);//每次都会多输出一个ASII为10的换行。因为每次循环时候,在检测缓存的时候,发现有个‘\n’,便会浪费一次循环。 } }
例子2使用了空格控制符后:
#include <stdio.h> //#include <Windows.h> int main() { int i; char j; for(i = 0;i < 10;i++) { scanf(" %c",&j);/*这里%前有空格*/ //每一次循环,都是先将输入的字符给j,然后再将enter的‘\n’给j. printf("j=%c\n",j); } }
会发现,例子2中输出的并没有换行,‘\n’。
4、处理scanf()函数误输入造成程序死锁或出错。
例如以下的程序:
#include <stdio.h> //#include <Windows.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> //#include <Windows.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()函数执行成功时的返回值是成功读取的变量数,也就是说,你这个scanf()函数有几个变量,如果scanf()函数全部正常读取, //它就返回几。但这里还要注意另一个问题,如果输入了非法数据,键盘缓冲区就可能还个有残余信息问题。 }
同样,可以看下下面的例子。
#include <stdio.h> //#include <Windows.h> int main() { char ch1,ch2; printf("Input for ch1%\n"); scanf("%c",&ch1); printf("ch1=%c\n",ch1); printf("Input for ch2\n"); scanf("%c",&ch2);//并不会等你输入ch2的值因为,ch1输入之后,还输入了enter,所以缓存中还有字符,所以取‘\n’赋给ch2。 printf("ch2=%c\n",ch2); }
scanf是从标准输入缓冲区中读取输入的数据,而%c的字符输入格式会接收回车字符,在输入第一个scanf时输入字符后按回车结束,输入缓冲中保存了这个回车符,遇到第二个scanf时,它自动把这个回车符赋给了ch2 。而如果第二个scanf的输入格式不是%c时,由于格式不匹配,这个回车符会被自动忽略,所以只有在连续输入两个%c的格式时才会出现这样的问题!
如下面的情况:
#include <stdio.h> //#include <Windows.h> int main() { char ch1; int ch2; printf("Input for ch1%\n"); scanf("%c",&ch1); printf("ch1=%c\n",ch1); printf("Input for ch2\n"); scanf("%d",&ch2);//并不会等你输入ch2的值因为,ch1输入之后,还输入了enter,所以缓存中还有字符,所以取‘\n’赋给ch2。 printf("ch2=%d\n",ch2); }