Bootstrap

C语言学习初级阶段(数据)——scanf读取标准输入

一、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在键盘输入的时候,输入一个数就按一个空格,目的就是区分输入的是不同的数,而不是一个数。

悦读

道可道,非常道;名可名,非常名。 无名,天地之始,有名,万物之母。 故常无欲,以观其妙,常有欲,以观其徼。 此两者,同出而异名,同谓之玄,玄之又玄,众妙之门。

;