scanf的缓冲区问题

昨天在写其他代码时,突然使用了一下scanf函数,结果把自己用死了。

    int a;
    int b;
    char op;
    printf("输入两个数:");
    scanf("%d%d", &a,&b);
    printf("输入操作方法:");
    scanf("%c", &op);
    printf(" %d %c %d = ", a, op, b);

我的输入格式:
1 2↙
当我输完两个操作数,按下回车键,准备输入操作方法程序却变成了这。
这里写图片描述

这就是典型的scanf缓冲区问题。

当输入这句时1 2↙,输入缓冲区里存放着1□2\n,第一次scanf读走1 给 a,2 给 b,中间的空格丢弃,此时scanf已经读完。但是缓冲区里还遗留有一个 \n .当第二个scanf读入时首先检查缓冲区,发现缓冲区里还有 \n ,而且正好匹配%c,于是直接读走 \n 给 c。

这个问题我一开始就想到了缓冲区的问题,但是另一个问题又把我坑进去了。
我用 fflush(stdin);去刷新缓冲区。修改代码如下:

    int a;
    int b;
    char op;
    printf("输入两个数:");
    scanf("%d%d", &a, &b);
    printf("输入操作方法:");
    fflush(stdin);
    scanf("%c", &op);
    printf(" %d %c %d = \n", a, op, b);

这样我以为解决了缓冲区的问题,殊不知完全不是我想的那样。于是我又查阅了fflush

int fflush(FILE *stream);
如果stream指向输出流或者更新流(update stream),并且这个更新流最近执行的操作不是输入,那么fflush函数将把任何未被写入的数据写入stream指向的文件(如标准输出文件stdout)。否则,fflush函数的行为是不确定的。fflush(NULL)清空所有输出流和上面提到的更新流。如果发生写错误,fflush函数会给那些流打上错误标记,并且返回EOF,否则返回0。

我们不能用这种不确定的代码再我们的程序中。

正确解决缓冲区问题的方法

  1. 用rewind函数刷新缓冲区。
    rewind(stdin);
    这里写图片描述
    可以达到我们的预期。
    rewind函数是把指定流的读写指针重新指向开头。
    2.scanf(“%*c”); 读取缓冲区中指定格式的内容并且丢弃。
    这里写图片描述
    把缓冲区的\n读走并且丢弃。很明显可以达到预期
    3.while(getchar() != “\n” && getchar() != EOF);
    这里写图片描述

看了很多文章都说的用fflush(stdin)刷新输入缓冲区,其实是错误的。
我也是“实践出真知”,所以学知识一定要“求甚解”。
当然scanf还有很多奥妙的用法,很多文章都有介绍这里就不赘述了。

发布了94 篇原创文章 · 获赞 237 · 访问量 7万+

猜你喜欢

转载自blog.csdn.net/qq_38646470/article/details/80936905