
本文深入探讨java `scanner`类的`hasnext()`方法,解释了它在不消费输入时导致无限循环的原因。通过对比`for`循环和`while`循环中`hasnext()`的不同应用,强调了及时消费输入的重要性,并提供了避免此类常见陷阱的编程实践。
理解Scanner.hasNext()的工作原理
在Java编程中,java.util.Scanner类是处理用户输入或文件内容的重要工具。它提供了一系列方法来检查和读取不同类型的输入。其中,hasNext()方法用于检查输入流中是否还有下一个标记(token)可用。关键在于,hasNext()仅仅是进行“检查”,它并不会实际“消费”或移除输入流中的任何数据。这意味着,如果输入流中有一个或多个标记等待被读取,hasNext()将返回true;反之,如果输入流已到达末尾或者没有可用的标记,它将返回false。
hasNext()导致无限循环的场景分析
考虑以下代码片段,它试图使用for循环和hasNext()来处理输入:
import java.util.Scanner;
public class Vocabulary {
public static void main(String[] args) {
Scanner standardInput = new Scanner(System.in);
for(int i = 0; standardInput.hasNext(); i++){
System.out.print(i);
}
// standardInput.close(); // 实际应用中应关闭Scanner
}
}当运行这段代码并从控制台输入一些内容(例如,键入“hello”并按回车)时,程序会进入一个无限循环。循环体内的System.out.print(i)会不断打印递增的整数i。这是因为standardInput.hasNext()的条件始终为true。
原因解析:
立即学习“Java免费学习笔记(深入)”;
- 用户在控制台输入“hello”后,这个字符串作为一个标记(token)进入了System.in的输入缓冲区。
- standardInput.hasNext()方法检测到缓冲区中存在“hello”这个标记,因此返回true。
- 循环继续执行,打印当前的i值。
- 然而,在循环体内部,我们没有调用任何Scanner的读取方法(例如next()、nextLine()、nextInt()等)来实际消费或移除“hello”这个标记。
- 由于“hello”始终存在于输入缓冲区中,standardInput.hasNext()在下一次迭代时仍然会检测到它,并继续返回true。
- 这个过程无限重复,导致程序陷入死循环。
正确使用hasNext()与消费输入
为了避免无限循环,在使用hasNext()检查输入可用性之后,必须在循环体内调用相应的next()系列方法来消费输入。以下是一个正确处理输入的示例,它会计算用户输入的所有数字的总和:
import java.util.Scanner;
public class Sum {
public static void main(String[] args) {
Scanner standardInput = new Scanner(System.in);
double sum = 0;
// 循环条件检查是否有下一个double类型的数字
while(standardInput.hasNextDouble()) {
double nextNumber = standardInput.nextDouble(); // 消费并获取下一个double
sum += nextNumber;
}
System.out.println("The Sum is " + sum + ".");
standardInput.close(); // 关闭Scanner以释放资源
}
}工作原理:
- while(standardInput.hasNextDouble())检查输入流中是否有一个可被解析为double类型的标记。
- 如果存在,条件为true,循环体执行。
- double nextNumber = standardInput.nextDouble();这行代码消费(即读取并移除了)输入流中的下一个double类型标记。
- 该数字被加到sum变量中。
- 循环继续,再次检查hasNextDouble()。如果用户输入了更多数字,则继续消费;如果用户输入了非数字内容或者通过特定的终止符(如Unix/Linux/macOS上的Ctrl+D,Windows上的Ctrl+Z)表示输入结束,hasNextDouble()最终将返回false。
- 当hasNextDouble()返回false时,循环终止,程序打印出总和。
注意事项与最佳实践
- 检查与消费配对: 始终记住hasNext()系列方法仅用于检查,next()系列方法才用于消费。两者应配对使用,以确保输入流能够被正确处理和推进。
-
输入终止符: 对于从System.in读取的用户输入,通常需要一个特定的组合键来模拟输入流的结束,以便hasNext()最终能返回false。
- 在Unix/Linux/macOS系统上,通常是按下Ctrl + D。
- 在Windows系统上,通常是按下Ctrl + Z,然后按回车。
- 关闭Scanner: 在不再需要Scanner对象时,务必调用standardInput.close()方法来关闭它,以释放底层系统资源(例如文件句柄或输入流)。这有助于避免资源泄漏。
- 选择合适的hasNextXxx()方法: 根据预期输入的类型,选择最合适的hasNextXxx()方法(如hasNextInt()、hasNextLine()、hasNextBoolean()等),以确保类型匹配和程序的健壮性。
总结
Scanner.hasNext()方法是Java中检查输入流的重要工具,但其作用仅限于检查而非消费。为了避免无限循环并确保程序正确处理输入,开发者必须在循环体内显式地调用Scanner的next()系列方法来消费检测到的输入标记。理解“检查”与“消费”之间的区别,是有效使用Scanner并编写健壮输入处理代码的关键。










