感谢大家分享,今天在查阅了一些资料之后,终于直观的搞懂了C语言中的scanf的工作机制,以及经常遇到的各种奇葩问题。
了解其工作机制,才能知晓其外表下的内在原因,总是有道理的。
scanf()函数的工作过程可以简单描述为如下过程:
只有当输入回车键的时候,scanf()才开始工作。
文字描述为:
为了提高存储器访问效率,操作系统的缓冲区管理机制,键盘输入的数据不是立即为scanf()读取,而是暂存于内存缓冲区,只有敲下回车键的时候,scanf()函数才开始工作(尤其要注意的是,回车键(\n,ASCII为10,换行为\r,ASCII为13)也会被送入缓冲区)。之后,scanf()从缓冲区中读取数据,只有当缓冲区为空时,程序才会停下来,等待用户输入;反之,缓冲区不为空时,scanf()可以一直按照格式说明读取数据。
int scanf(格式说明,地址列表);
第一个参数:输入数据的格式要求,参见示例2
格式列表中,可能有多个数据格式,一般情况下,输入数据使用空格(space)、tab、回车进行分割;但是,当输入格式为%c时,空格、tab、回车会以ASCII码形式存储到对应参数地址所在内存空间。
第二个参数:读取的数据将要存入的地址,需要是变量的地址
scanf()的返回值:
1. 正整数,正确读取数据的个数
2. 0,输入数据与要求数据格式不符合,即正确读取数据个数为0.
3. EOF,在stdio.h中定义的常量,值一般为-1,Windows输入CTRL+Z。
char ch,s1[20],s2[20];
scanf("%s",s1);
scanf("%s",s2);
scanf("%c",&ch);
printf("s1=%s,s2=%s,ch=%d",s1,s2,ch);
int a,b;
scanf("%d,%d",&a,&b);
printf("a=%d,b=%d",a,b);
//input:1,2回车或1空格2回车,output:a=1,b=2
int a,b;
scanf("a=%d,b=%d",&a,&b);
printf("a=%d,b=%d",a,b);
//此时只有输入a=1,b=2才合理,仿照上面的输入会得到a=随机数,b=随机数
说明:
scanf(“a=%d,b=%d”, &a,&b)语句为例,格式说明参数为字符串“a=%d,b=%d”,其中“%d”为格式控制符,“a=”和“,b=”分别为第一、二个待接收数据的前驱字符(很多书称做格式说明参数中的“其它字符”);“ a”和“ b”为地址参数。
注意:在进行第二步数据类型匹配的时候(即匹配%d的时候),允许在对应数据之前输入0个或多个空格、回车、Tab键,scanf会自动忽略它们。
因此,输入:a=1,b=2↙或a=□1,b=□2↙效果是一样的,都会让a值为1,b值为2。
scanf函数对上面输入数据的匹配过程如下:
1)将前驱字符“a=”与缓冲区的数据进行匹配,完全相同;
2)进行数据类型“%d”的匹配,如果这时缓冲区内有空格之类分隔符,忽略它们,检测到整型数据1,类型合法(根据1后面的非整型数据逗号判断出整型数据输入结束),将1写入变量a的空间;
3)继续进行下一轮匹配,先对照第二个数据的前驱字符“,b=”,完全相等;
4)按格式说明参数匹配“%d”,忽略空格,接收数据2,存入对应空间;
5)检测到格式说明参数字符串已达到末尾,结束工作。
但是,如果输入:□a=1,□b=2↙则在scanf()函数测试到第一个输入字符空格的时候就认为它不匹配格式参数中要求的“a=”,立刻停止工作。因此变量a、b并没有被写入数据,它们的值是随机数(a、b是局部变量)。
再如,输入:a=1,□b=2↙则a值为1,b值为一个随机数。因为当检测到“,□b=”与格式说明参数中的前驱字符“,b=”不匹配时,scanf()函数结束工作,用户输入的整数2并未被识别进而存入变量b中。
说明:若格式说明参数中有空格,将自动被忽略。如语句scanf(”□%d□%d”, &a, &b)等价于scanf(“%d%d”, &a, &b),可以认为没有“前驱字符”,匹配流程跳过第一步。
int a[10];
int i=0;
while(scanf("%d",&a[i++])!=EOF){}