一、scanf函数的原理
- 输入和输出是通过标准函数库实现的,C语言通过scanf函数读取键盘输入,键盘输入又称标准输入。
- 当scanf函数读取标准输入时,如果还没有输入任何内容,那么scanf函数会被卡住(专业用于为阻塞)。
- scanf输入的其实全部都是字符串,要把输入的字符串转换成相应的格式类型。scanf(“%d”,&i); scanf里面的前后类型要一致
1.1、自己理解
#include <stdio.h>
//scanf用来读取标准输入,scanf把标准输入的内容存放到某个变量空间,因此变量必须取地址
int main(){
int i;
char c;
scanf("%d",&i);
printf("i=%d\n",i);
scanf("%c",&c);
printf("c=%c\n",c);
return 0;
}
输出结果:
为什么标准输入C,但是为什么空了2行呢?
因为第一个scanf输入相当于是一个字符串,实际上输入的是10\n,但是%d只会读取0-9的字符,不会读取\n,所以第一printf打印的是10
第二个printf打印的是\n,因为%c取的是字符串,\n是一个短义字符,所以%c会读取它。
因为scanf标准输入的时候,必须按回车才能生效。一个回车相当于一个\n,所以会打印2个\n。
如何解决这个问题呢?
在两个scanf之间加入 fflush(stdin);//清空标准输入缓冲区
即可
运行结果如下:
1.2、王道解释
行缓冲:在这种情况下,当在输入和输出中遇到换行符时,将执行真正的IO处理操作,这时,我们输入的字符先存放到缓冲区中,等按下回车键换行时才进行实际的I/O操作.
典型代表是标准输入缓冲区(stdin)和标准输出缓冲区(stdout) , printf 使用的是stdout 。
如上面的例子所示,我们向标准输入缓冲区中放入的字符为’20\n’,(scanf输入的都是字符串,字符串后面都会有\n)输入’\n’(回车)后, scanf函数才开始匹配, scanf函数中的%d 匹配整型数20,然后放入变量i中,接着进行打印输出,这时’\n’仍然在标准输入缓冲区(stdin)内,如果第二个scanf函数为scanf(“%d”,&i),那么依然会发生阻塞
scanf函数匹配一个字符时,会在缓冲区删除对应的字符.因为在执行scanf(“%c”,&c)语句时,不会忽略任何字符,所以scanf(“%c”,&c)读取了还在缓冲区中残留的’\n’.
#include <stdio.h>
//scanf用来读取标准输入,scanf把标准输入的内容存放到某个变量空间,因此变量必须取地址
int main(){
int i;
char c;
float f;
scanf("%d",&i);
printf("i=%d\n",i);
// fflush(stdin);//清空标准输入缓冲区
// scanf("%c",&c);
// printf("c=%c\n",c);
scanf("%f",&f);
printf("f=%f",f);
return 0;
}
输出结果:
20
i=20
98.5
f=98.500000
此结果,输入20之后,生成了i=20,其次输入98.5,在生成f=98.500000。发生了阻塞
为什么scanf读取浮点数的时候,在两个scanf之间不用fflush(stdin)来清空标准输入缓冲区呢?
因为scanf函数在读取整型数、浮点数、字符串(后面介绍数组时讲解字符串)时,会忽略’\n’(回车符)、空格符等字符〈忽略是指scanf函数执行时会首先删除这些字符,然后再阻塞)
解释:
当标准输入20的时候,相当于一个字符串,20\n就存入到了标准输入缓冲区,然后%d会读取20,此时标准输入缓冲区内剩下了\n,%f开始读取\n,但是%f是读取浮点数,\n是转义字符,遇到它的时候,就会忽略情况\n。
为什么会发生阻塞卡住?
因为标准输入缓冲区内为空,所以会卡住。不空,则不会发生阻塞卡住。
1.3、注意
#include <stdio.h>
//scanf一次读取多种类型
int main(){
int i,ret;
char c;
float f;
ret=scanf("%d %c%f",&i,&c,&f);//ret是scanf匹配成功的个数
printf("i=%d,c=%c,f=%5.2f\n",i,c,f);
return 0;
}
在scanf输入多个的时候,如果%c在中间一定要在%c前面加一个空格,如果%c在开头就不要加空格。
scanf在键盘输入的时候,输入一个数就按一个空格,目的就是区分输入的是不同的数,而不是一个数。