0


关于scanf读入的细节问题

主要是过程和细节

1.连续输入两个scanf的问题(字符)

1.1问题

连续使用scanf输入有个坑,最后的回车是会留在缓冲区的,会被下面的%c或者%s吃掉,最好后面加getchar()吃掉多余的\n。

如果输入了第一个或第一串字符后,按了回车键(有些编译器点运行其实也是键入了回车的)
回车键相当于\n,也算字符,它不是没输出,是输出了一行空白的,看起来好像没输出一样。

1.2解决办法

  1. 加空格
  2. 加getchar()吸收回车
  3. 加fflush(stdin) 或 rewind(stdin) 来清空键盘缓冲区
清空缓冲区:
setbuf(stdin,NULL);(这个windows和linux下都可以)
fflush(stdin);(这个只能windows);
1.在循环语句输入字符类型数据时也要及时接收回车键或者清空
键盘缓冲区
2.二者使用意义不同,getchar()是接收缓冲区多余的回车,而
fflush(stdin)和rewind(stdin)是清除缓冲区
空格作用

scanf("%d%c"…)

scanf("%d %c"…)
不同。
后者读 %c 将跳过 第一个整数值后面的 所有的 空白符号 (空格,\t,\n) 也就是说,那个缓冲区的回车直接跳过了。
前者%c会读入根随第一个整数值后面的第一个字符,无论是一般字符还是空白符号,回车就会读进去(如果下一个连续scanf读入就不好,就会读到回车)


举个小例子说明一下下面这个问题:

2.scanf读入过程细节(不限)

第一个:

输入:345abc
输出:345345
#include<stdio.h>voidmain(){int a;scanf("%d",&a);printf("%d",a);scanf("%d",&a);printf("%d",a);}

scanf读入数据的方式是挨个读入,第一个程序scanf先读3,然后尝试读下一位,发现之前读的可以和下一位组成新的和

类型相符合

的值,再尝试读下一位,这种策略叫做贪心策略。在读完345后,scanf发现a不能和345组成新的与类型相符合的值,于是scanf退出,345被赋值给a。此时缓冲区只剩abc三个字符,下一个scanf开始读入,因为转义符号是%d,这说明scanf应该读入int型值,但是a为字符型,一开始就不符合,scanf退出。此时a没有被赋值,还是345。所以两次输出都是345。

第二个:

#include<stdio.h>voidmain(){int a,char c;scanf("%d",&a);printf("%d",a);scanf("%c",&c);printf("%c",c);}
输入:345abc
输出:345a

第二个程序,并不是说把打印的类型换了缓冲区就变了,而是scanf收到的信号不同,读入方式不同而导致的。345同理,然后缓冲区剩abc,%c代表scanf只需读入一个字符即可,刚好a满足条件,读入a,scanf退出。所以第二个程序输出为345a。

对于要输入

I love you

呢?
scanf输入字符串(%s)遇到空格会结束
就用gets等可以忽略空格的东西输入字符串

3.为什么读入整数没有那么多事?

3.1scanf()函数输入字符时的执行原理

1、用户输入的字符,会以ASCII码的形式存储在键盘缓冲区;
2、scanf()函数每执行一次,就从键盘缓冲区读出一个ASCII值;
3、调用scanf()函数时,用户最后输入的回车键也会以换行符

'\n'

的ASCII码储存在键盘缓冲区;

3.2读取整数时

如果输入整数或者字符串时,为什么又不读取回车键的ASCII值呢?

答:scanf()函数中的%d和%c分别限定了输入的数据类型,如果在

%d

的限定下输入了回车、空格或者Tab,是不会将其转换为ASCII值存入缓冲区,scanf()会继续等待用户输入整数并读取到整型数据后,它才会结束,这也是我们为什么输入多个整数是可以用空格、回车以及Tab等隔开多个整型数据的输入的原因。

标签:

本文转载自: https://blog.csdn.net/qq_46489944/article/details/119856039
版权归原作者 Bluerain. 所有, 如有侵权,请联系我们删除。

“关于scanf读入的细节问题”的评论:

还没有评论