【Java实战经验】Scanner输入异常提示重新输入导致死循环的处理

在使用 Scanner 获取输入时,经常出现输入的类型不匹配而导致的程序异常,直接停止程序很不友好,所以我们需要对异常进行处理

初步解决

不难想到,应该使用一个死循环接收用户输入,如果正确则跳出,否则捕获异常提示重新输入,写出如下代码:

Scanner scan = new Scanner(System.in);
int a;

System.out.print("请输入一个整数:");
while(true)
{
    try
    {
        a = scan.nextInt();
        break;
    } catch (Exception e) {
        System.out.print("输入了不合法的整数,请重新输入:");
    }
}

System.out.println("你输入的整数为:" + a);

但运行结果却不是我们想的那样子:
在这里插入图片描述

分析

在使用 nextInt() 输入非法整数时,异常被捕获
缓存区的内容并没有被读取出来
直接导致了下一次循环 nextInt() 直接读取了缓存中的非法整数,再次触发异常被捕获,由此导致死循环。

所以,在异常捕获那一块应该将缓存清空,以进行下一次阻塞输入

最终方案

只需小小改动 catch 中的内容,让缓存区中的内容读取出来(相当于清除)

Scanner scan = new Scanner(System.in);
int a;

System.out.print("请输入一个整数:");
while(true)
{
    try
    {
        a = scan.nextInt();
        break;
    } catch (Exception e) {
        System.out.print("[ "+scan.nextLine()+" ] 不是合法的整数\n请重新输入:");
    }
}

System.out.println("你输入的整数为:" + a);

运行结果:
在这里插入图片描述
联系总结:
其实在 C 语言也遇到了同样的问题:scanf 输入异常导致死循环
只在 Java 中提供了异常捕获机制,使得整个异常的处理变得非常简单,而 C 中需要人为地根据 scanf 的返回值去判断是否有输入异常,灵活度也就打打降低了
上面代码中 catch 能捕获一切异常,因为 Exception 是一切异常的父类,这就使得其不仅能捕获这种特定的异常,其他异常也能通通捕获,不用人为地判断是那种异常,灵活度高。


while(true)
{
	try
	{
		MyScanner mscan = new MyScanner();
		mscan.nextStudy();
	} catch(Exception lazy)
	{
		System.out.print("[ "+mscan.nextDo()+" ] 不是合法的动作\n请重重新调整:");
	}
}

(寒冰小澈)

posted @ 2021-02-12 09:07  IceClean  阅读(134)  评论(0)    收藏  举报