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);
}