Do ..虽然循环不会停止以等待uinput

问题描述

我正在通过基本的输入有效性检查机制实现超简单的菜单选择过程。合法输入为{1,2,3},因此可能出现的问题可能是超出此范围的数字,或者是非整数。我的代码如下所示。这对于前一个问题(即,当我输入“ 4”时)很好用,但是对于后一个问题(当我尝试输入字符时),它一遍又一遍地打印无效消息,而不是等待新的输入,就像这样除第一次迭代外,所有迭代均完全跳过scanf行。我在这里缺少什么?

do{
      try = scanf("%d",&selection);
      if(try!=1 || selection < 1 || selection > 3){
           printf("\nInvalid input. Dear guest,please enter '1','2',or '3'.\n\nInput:");
      }
 }while(try!=1 || selection < 1 || selection > 3);

解决方法

在输入了不能作为十进制数字的文本表示形式的一部分的字符(例如'A')之后,程序可用的输入为字节序列'A''\ n'(后者是换行符),在Windows上可能是'A''\ r''\ n'(回车后跟换行符)。

当scanf尝试从这些字符中解析一个数字时,它已经在'A'处取消了并将其放回到输入流中。 C中的输入流确保您可以执行至少一个{ {1}},也就是说,将至少一个字符放回流中,这样它将成为下一个输入操作读取的第一个字符。这种简单但巧妙的功能使处理可变格式输入变得很容易:想象一下,您解析了一些C源代码中的表达式,并且在语法上允许使用整数文字或变量名称作为下一个标记。 :您可以先尝试该数字,如果失败,输入仍包含所有要处理的变量名称。记住“第一个失败字符”并将其提供给程序的其他部分的工作封装在ungetc()实现中。

这就是这里发生的情况。第一个失败的scanf()将“ A”放回,以便下次尝试 ad infinitum 再次遇到它。必须从输入中删除“ A”。更具体地说,应该从输入中完全删除下一个“单词”:用户可能已经输入“ kkjkllkjlk”,并且您不希望为此输入10条错误消息。您可以决定是否接受“ lklkj2”(并阅读2),但是丢弃整个单词更简单。

您还可以决定是否接受“ 1 2 3 2”作为4个有效的连续输入,或者是否需要在数字之间输入换行符。出于普遍性(例如,如果输入不是来自终端),我将接受所有用任何类型的空格分隔的数字序列。在这种情况下,您只想阅读下一个空格:

FILE

这应该可以解决问题。可能会有更多的空白,包括换行符等,但这没关系:scanf的符号输入转换(如%d)跳过前导空白。

我认为通过结束输入(通过在Windows控制台中按Ctrl-z或在Posix终端上按Ctrl-d来插入文件结尾)来让用户退出程序是很整洁的,所以我将进行测试对于scanf()返回EOF的特殊情况。如果输入来自管道,则可能实际上是必不可少的。即使在丢弃错误输入的代码中,输入错误后紧接着EOF的边缘情况也需要检查EOF(Posix:#include <ctype.h> // ... while(!isspace(getchar())) { /* ignore */ } 会挂起; -n取消了通常用echo -n "a" | myprog附加的换行符) 。综上所述,我对输入循环的看法是:

echo

相关问答

依赖报错 idea导入项目后依赖报错,解决方案:https://blog....
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下...
错误1:gradle项目控制台输出为乱码 # 解决方案:https://bl...
错误还原:在查询的过程中,传入的workType为0时,该条件不起...
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct...