scanf读取标准输入缓冲区的原理
代码如下:
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
//原理对于考研初试不重要,机试很重要
//缓冲区为空的时候,scanf才会阻塞
int main()
{
int i;
char c;
float f;
scanf("%d", &i);
printf("i=%d\n", i);
scanf("%c", &c);
printf("c=%c\n", c);
scanf("%f", &f);
printf("f=%f\n", f);
scanf("%c", &c);
printf("c=%c\n", c);
}
运行截图如下:
原理:
C语言中的scanf函数的作用是从标准输入设备(通常是键盘)读取输入值,并存储到参数列表中指针所指向的内存单元。
1、scanf的返回值
scanf通常返回的是成功赋值(从标准输入设备赋值到参数列表所指定的内存区域)的数据项数,如果出错或是遇到end of file(注意,如果想从键盘输入EOF,在windows的DOS窗口输入3次Ctrl+Z),则返回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对不同类型输入的处理方式
3.1 整数%d
对于整型数据的输入,也就是说"%d"类型的输入,scanf默认的分割符是所有的空白字符(空格(space),回车(enter)和指标符(table)都行)。也就是说如果一个scanf函数中出现scanf(“%d%d”,&x,&y),那么用任何一个空白字符来分隔两个整数a,b的值,变量a,b都可以接收到正确的输入。另外,要注意的是,scanf对于数字输入,会忽略输入数据项前面的空白字符。
3.2 浮点型%f
与整型%d同理。
3.3 字符串%s
scanf对于字符串输入的处理和对整数类似,会忽略前面的空白字符,而且默认的分隔符是所有的空白字符。但是,要注意的是,由于C语言中,没有string类型,都是用char型数组来表示。因此,scanf会为每一个输入的字符串最后加一个‘\0’,对于scanf的边界控制还是要小心。
3.4 字符%c
scanf在处理对字符数据的输入时,既不会忽略前面的空白字符,默认也没有任何分隔字符。所有的字符,包括空白字符都会被当成输入字符。从上面的运行截图可以看出,在对字符数据输入的时候,由于缓冲区中有回车空格等数据,会导致输入数据比较诡异。
当我们输入整型数据1后按回车的时候就会把整型数据1连同换行符送入缓冲区,然而scanf只会接收整型数据1,那么缓冲区内就只有一个换行符了。当scanf处理字符数据的时候,因为缓冲区内非空,那么scanf直接读取缓冲区中的换行符所以输出的结果就是一个换行罢了。当我们输入浮点型数据1.5后按回车的时候也会把浮点型数据1.5连同换行符送入缓冲区,同样scanf只会接收浮点型数据1.5,缓冲区内还会剩下一个换行符。当scanf处理字符数据的时候,因为缓冲区内非空,那么scanf直接读取缓冲区中的换行符所以输出的结果还是一个换行。