本文深入探讨java `scanner`类的`hasnext()`方法,解释了它在不消费输入时导致无限循环的原因。通过对比`for`循环和`while`循环中`hasnext()`的不同应用,强调了及时消费输入的重要性,并提供了避免此类常见陷阱的编程实践。
在Java编程中,java.util.Scanner类是处理用户输入或文件内容的重要工具。它提供了一系列方法来检查和读取不同类型的输入。其中,hasNext()方法用于检查输入流中是否还有下一个标记(token)可用。关键在于,hasNext()仅仅是进行“检查”,它并不会实际“消费”或移除输入流中的任何数据。这意味着,如果输入流中有一个或多个标记等待被读取,hasNext()将返回true;反之,如果输入流已到达末尾或者没有可用的标记,它将返回false。
考虑以下代码片段,它试图使用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。
原因解析:
没有调用任何Scanner的读取方法(例如next()、nextLine()、nextInt()等)来实际消费或移除“hello”这个标记。为了避免无限循环,在使用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以释放资源
}
}工作原理:
Scanner.hasNext()方法是Java中检查输入流的重要工具,但其作用仅限于检查而非消费。为了避免无限循环并确保程序正确处理输入,开发者必须在循环体内显式地调用Scanner的next()系列方法来消费检测到的输入标记。理解“检查”与“消费”之间的区别,是有效使用Scanner并编写健壮输入处理代码的关键。