在使用 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请重重新调整:");
}
}
(寒冰小澈)