C语⾔中scanf函数与空格回车的⽤法说明
众所周知,C语⾔中的scanf函数的作⽤是从标准输⼊设备(通常是键盘)读取输⼊值,并存储到参数列表中指针所指向的内存单元。
下⾯从⼏个⽅⾯说⼀下⼀些稍微细节的东西。下⾯的实验都在vc6.0中通过。
1、scanf的返回值
scanf通常返回的是成功赋值(从标准输⼊设备赋值到参数列表所指定的内存区域)的数据项数,如果出错或是遇到end of
file(注意,如果想从键盘输⼊EOF,在windows的DOS窗⼝⽤Ctrl+Z 或F6;在UNIX系统上,⽤CTRL+D。),则返回EOF,⽐如:
scanf("%d%d", &x, &y);
如果x和y都被成功读⼊,那么scanf的返回值就是2;
如果只有x被成功读⼊,返回值为1;
如果x和y都未被成功读⼊,返回值为0;
如果遇到错误或遇到end of file,返回值为EOF。
2、scanf的处理机制
scanf以删除的⽅式从缓冲区读⼊数据(来⾃标准输⼊设备的数据存储在缓冲区),也就是说,scanf从缓冲区读⼊⼀个数据项,该数据项在缓冲区中就被清除掉了。
⽽如果scanf需要读取⼀个数据项,返现缓冲区当前是空的,那么程序就会在scanf代码处阻塞,等待⽤户输⼊,scanf函数接收到相应的数据项之后,在缓冲区中将这⼀数据项清除,scanf函数返回,程序继续执⾏。
3、scanf对不同类型输⼊的处理⽅式
⾸先,要清除⼀个概念:空⽩字符(white space)。⼀般,程序中所指的空⽩字符是指空格(space),回车(enter)和指标符(table)。
3.1 整数%d
对于整型数据的输⼊,也就是说"%d"类型的输⼊,scanf默认的分割符是所有的空⽩字符(空格,回车和指标符都⾏)。
也就是说如果⼀个scanf函数中出现scanf("%d%d",&a,&b),那么⽤任何⼀个空⽩字符来分隔两个整数a,b的值,变量a,b都可以接收到正确的输⼊。
另外,要注意的是,scanf对于数字输⼊,会忽略输⼊数据项前⾯的空⽩字符。下⾯是例1:
Code:
#include<stdio.h>
int main()
{
int a,b;
printf("Input the value of a and b:");
while(scanf("%d%d",&a,&b)!=EOF)
{
printf("a=%d,b=%d\n",a,b);
printf("Input the value of a and b:");
}
return 0;
}
Output:
Input the value of a and b:123 456
a=123,b=456
Input the value of a and b:123 456
a=123,b=456
Input the value of a and b:123
456
a=123,b=456
Input the value of a and b:
123 456
a=123,b=456
Input the value of a and b: 123 456
a=123,b=456
Input the value of a and b: 123 456
a=123,b=456
Input the value of a and b:^Z
Press any key to continue
3.2 字符串%s
scanf对于字符串输⼊的处理和对整数类似,会忽略前导的空⽩字符,⽽且默认的分隔符是所有的空⽩字符。但是,要注意的是,由于C语⾔中,没有string类型,都是⽤char型数组来表⽰。
因此,scanf会为每⼀个输⼊的字符串最后加⼀个‘\0'。下⾯是⼀个例⼦,可以看出scanf这货的边界控制还是要⼩⼼。
如下例2。
#include<stdio.h>
int main()
{
char a[5],b[5];
int i;
printf("Input the value of a and b:");
while(scanf("%s%s",a,b)!=EOF)
{
printf("a=%s,b=%s\n",a,b);
for(i=0;i<5;i++)
printf("%d:(%c) ",a[i],a[i]);
printf("\n");
for(i=0;i<5;i++)
printf("%d:(%c) ",b[i],b[i]);
printf("\n");
printf("Input the value of a and b:");
}
return 0;
}
运⾏结果:
3.3 字符%c
scanf在处理对字符数据的输⼊时,既不会忽略前导空⽩字符,默认也没有任何分隔字符。所有的字符,包括空⽩字符都会被当成输⼊字符。
下⾯是例3。
#include<stdio.h>
int main()
{
char a ,b ;
printf("Input the value of a and b:");
while(scanf("%c%c",&a,&b)!=EOF)
{
printf("a=%c,b=%c\n",a,b);
printf("Input the value of a and b:");
}
return 0;
}
运⾏结果:
可以看出,在对字符数据输⼊的时候,由于缓冲区中有回车空格等数据,会导致输⼊数据⽐较诡异,为了解决这个问题,有以下⽅法:
(1) 清空缓冲区
在微软系统中,有⼀个名为fflush(stdin)的函数,可以⽤来清空缓冲区,
如下例4。
#include<stdio.h>
int main()
{
char a ,b ;
printf("Input the value of a and b:");
while(scanf("%c%c",&a,&b)!=EOF)
{
printf("a=%c,b=%c\n",a,b);
fflush(stdin);
printf("Input the value of a and b:");
}
return 0;
}
运⾏结果:
(2)将缓冲区的数据读出来
有的编译系统并没有定义stdin的fflush操作,这个时候,可以把缓冲区中的数据读出来,有如下⼏种可⾏的⽅法:
1) getchar()
将例4中的fflush(stdin);语句换成
char c;
while((c=getchar())!='\n'&&c!=EOF);
运⾏效果和上⾯的相同。
2) gets()
char* gets(char* buffer)从stdin流中读取字符串,直⾄接受到换⾏符或EOF时停⽌,并将读取的结果存放在buffer指针所指向的字符数组中。换⾏符不作为读取串的内容,读取的换⾏符被转换为null值,并由此来结束字符串。
c语言scanf能输入中文吗读⼊成功,返回与参数buffer相同的指针;读⼊过程中遇到EOF(End-of-File)或发⽣错误,返回NULL指针。
所以在遇到返回值为NULL的情况,要⽤ferror或feof函数检查是发⽣错误还是遇到EOF。
要注意的是gets函数可以⽆限读取,不会判断上限,所以应该确保buffer的空间⾜够⼤,以便在执⾏读操作时不发⽣溢出。如果溢出,多出来的字符将被写⼊到堆栈中,这就覆盖了堆栈原先的内容,破坏⼀个或多个不相关变量的值。
将例4中的fflush(stdin);语句换成
char c[10];
gets(c);
运⾏效果也和上⾯的相同。
4、在stackoverflow上看到的⼀个问题
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
char *method1(void)
{
static char a[4];
scanf ("%s\n", a);
return a;
}
int main(void)
{
char *h = method1();
printf ("%s\n", h);
return 0;
}
运⾏结果:
ab
cd
ab
Press any key to continue
可以发现,输如两次之后才会输出。这个现象⽐较诡异,原因如下:
White space (such as blanks, tabs, or newlines) in the format string match any amount of white space, including none, in the input. Everything else matches only itself.
Thus with scanf ("%s\n", a) it will scan for a string followed by optional white space. Since after the first newline more whitespace may follow, scanf is not done after the first newline and looks what's next. You will notice that you can enter any number of newlines (or tabs or spaces) and scanf will still wait for more.
However, when you enter the second string, the sequence of whitespace is delimited and scanning stops.
以上这篇C语⾔中scanf函数与空格回车的⽤法说明就是⼩编分享给⼤家的全部内容了,希望能给⼤家⼀个参考,也希望⼤家多多⽀持。
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论