如何使用Scanner处理无效输入(InputMismatchException)导致的无限循环

问题描述 投票:0回答:5

所以,我被这段代码困住了:

import java.util.InputMismatchException;
import java.util.Scanner;

public class ConsoleReader {

    Scanner reader;

    public ConsoleReader() {
        reader = new Scanner(System.in);
        //reader.useDelimiter(System.getProperty("line.separator"));
    }

    public int readInt(String msg) {
        int num = 0;
        boolean loop = true;

        while (loop) {
            try {
                System.out.println(msg);
                num = reader.nextInt();

                loop = false;
            } catch (InputMismatchException e) {
                System.out.println("Invalid value!");
            } 
        }
        return num;
    }
}

这是我的输出:

插入整数:
无效值!
插入一个整数:
无效值!
...

java java.util.scanner infinite-loop
5个回答
74
投票

根据扫描仪的javadoc

当扫描仪抛出一个 输入不匹配异常,扫描仪 不会传递导致的令牌 例外,所以它可能是 通过其他方式检索或跳过 方法。

这意味着如果下一个标记不是

int
,它将抛出
InputMismatchException
,但标记仍保留在那里。因此,在循环的下一次迭代中,
reader.nextInt()
再次读取相同的标记并再次引发异常。你需要的是用完它。在你的
reader.next()
内添加一个
catch
来消耗令牌,该令牌是无效的,需要被丢弃。

...
} catch (InputMismatchException e) {
    System.out.println("Invalid value!");
    reader.next(); // this consumes the invalid token
} 

0
投票

我要做的是使用 Scanner.nextLine() 读取整行。然后创建另一个扫描仪来读取返回的字符串。

String line = reader.nextLine();
Scanner sc = new Scanner(line);

这将使您的示例函数如下所示:

  public int readInt(String msg) {
        int num = 0;
        boolean loop = true;

        while (loop) {
            try {
                System.out.println(msg);
                String line = reader.nextLine();
                Scanner sc = new Scanner(line);
                num = sc.nextInt();   
                loop = false;
            } catch (InputMismatchException e) {
                System.out.println("Invalid value!");

            } 
        }
        return num;
    }

这样,您就拥有一台扫描仪来获取输入,另一台扫描仪来验证输入,这样您就不必担心读者是否输入了正确的输入形式。


0
投票

while-do 的保护是“循环”变量。

在代码到达赋值循环 = false 之前抛出的异常本身; 准确地说,异常是在前面的语句中抛出的,即 num = reader.nextInt();

当抛出异常时,“loop”变量的值为“true”,但您的代码跳转到 catch 块,然后重复 while-do。这个 while-do 永远不会停止,因为下一次迭代将再次抛出异常,再次跳转到 catch 块等等。

要终止这个 while-do,您需要用另一个逻辑事物来保护您的 while-do,例如:

  1. 当读取器获取到非 int 字符时退出
  2. EOF时退出

这可以在 catch 块或其他一些行中完成。但精确的解决方案取决于您的规格。


0
投票

你也可以尝试这个:

   public int readInt(String msg) {
        int num = 0;
        try {
            System.out.println(msg);
            num = (new Scanner(System.in)).nextInt();
        } catch (InputMismatchException e) {
            System.out.println("Invalid value!");
            num = readInt(msg);
        } 
        return num;
    }

0
投票
package nzt.nazakthul.app;

import java.util.*;

public class NztMainApp {

    public static void main(String[] args) {
    ReadNumber readObj = new ReadNumber();
    readObj.readNumber();
    }

}

class ReadNumber {
int no;

    int readNumber() {
    Scanner number = new Scanner(System.in);
    int no=0;
    boolean b=true;
    do {

        try {
            System.out.print("Enter a number:\t");
            no = number.nextInt();
        } catch (InputMismatchException e) {
            System.out.println("No Number");
            //e.printStackTrace();

            b=false;
        }

    }

    while (b);
    return no;

    }

}

我个人使用 BufferedReader 和 InputStreamReader 来读取 String 并检查是否是数字,但是使用扫描仪可以减少代码。代码已检查并运行正常。

© www.soinside.com 2019 - 2024. All rights reserved.