
在java编程中,资源管理是确保程序健壮性和避免内存泄漏的关键一环。然而,对于`scanner`类,特别是当它包装`system.in`时,关于何时以及如何关闭它,存在一个普遍的误解。许多初学者,甚至一些经验不足的开发者,会认为既然打开了`scanner`,就应该在使用完毕后将其关闭,以遵循资源释放的原则。然而,对于`new scanner(system.in)`这种情况,这种做法是完全错误的,并且常常会导致运行时错误,例如`nosuchelementexception: no line found`。
要理解为何不应关闭Scanner(System.in),首先需要区分“资源”和“过滤资源”的概念。
问题的核心在于System.in。System.in是Java应用程序的标准输入流,它是一个由Java虚拟机(JVM)在启动时创建并管理的全局资源。它代表了控制台输入,通常是键盘。
Scanner(System.in)创建的Scanner对象包装了System.in。如果调用scanner.close(),它会尝试关闭底层的System.in流。一旦System.in被关闭,它就无法再次打开,这意味着整个应用程序将无法再从标准输入读取数据。这对于需要多次从用户获取输入的程序(如交互式菜单)来说是致命的。
关键原则:谁创建,谁关闭。 你没有创建System.in,它是JVM提供的。因此,你也不应该关闭它。只有当你创建了一个实际的资源(如FileInputStream),并且该资源由你负责管理其生命周期时,你才应该关闭它。IDE或静态代码分析工具有时会过度热心地提示你关闭Scanner,但它们可能无法区分Scanner包装的是否是System.in这种特殊情况。
对于那些确实需要关闭的资源,Java 7引入的try-with-resources语句是管理它们的最佳方式。它确保了资源在try块结束时(无论正常退出、return、`break还是抛出异常)都会被自动关闭,无需手动在finally块中编写关闭逻辑。
立即学习“Java免费学习笔记(深入)”;
示例:正确使用try-with-resources关闭文件输入流
import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
public class FileReaderExample {
public static void main(String[] args) {
// 对于需要关闭的资源(如文件读取器),使用 try-with-resources
try (BufferedReader reader = new BufferedReader(new FileReader("example.txt"))) {
String line;
while ((line = reader.readLine()) != null) {
System.out.println(line);
}
} catch (IOException e) {
System.err.println("读取文件时发生错误: " + e.getMessage());
}
// 注意:这里没有手动调用 reader.close(),它会在 try 块结束时自动完成
}
}重要提示: 尽管try-with-resources是资源管理的黄金标准,但不应将其应用于Scanner(System.in)。如果那样做,System.in仍会被关闭,导致后续输入不可用。
除了Scanner的关闭问题,原始代码中还存在一些可以优化的地方,以符合Java的编程规范和最佳实践。
Java社区遵循一套约定俗成的命名规范,这对于代码的可读性和团队协作至关重要。
遵循这些规范能让你的代码更易于他人理解和维护。
原始代码中的main(null);是一个严重的逻辑错误。在方法内部递归调用main方法会导致程序进入无限循环,每一次调用都会在调用栈上创建一个新的栈帧。最终,这会导致StackOverflowError,程序崩溃。
错误的递归调用示例:
public static void ms_calc () {
// ... 代码逻辑 ...
scanbpm.close(); // 错误关闭 System.in
main(null); // 致命的递归调用
}正确的方法是使用循环来重复程序的某个功能,例如一个菜单系统。
示例:使用循环实现重复功能
import java.util.InputMismatchException;
import java.util.Scanner;
public class BpmConverter {
public static void main(String[] args) {
// 创建一次 Scanner,并在整个程序生命周期中复用
// 不关闭 System.in 对应的 Scanner
Scanner scanner = new Scanner(System.in);
boolean running = true;
while (running) {
msCalc(scanner); // 将 scanner 传递给处理方法
System.out.println("\n是否继续计算?(yes/no)");
String choice = scanner.nextLine().trim().toLowerCase();
if (!choice.equals("yes")) {
running = false;
}
}
System.out.println("程序结束。");
// 在 main 方法结束时,System.in 会由 JVM 自动管理,无需手动关闭
}
public static void msCalc(Scanner scanner) { // 接收 Scanner 作为参数
System.out.println("What is your BPM?");
int bpm = getUserInt(scanner, 1, 2300);
// 计算逻辑
double mS1 = Math.round((bpm * 100.0) / 6) * 4; // 使用 100.0 确保浮点运算
double mS2 = Math.round((bpm * 100.0) / 6) * 2;
int mS1R = (int) mS1;
int mS2R = (int) mS2;
System.out.println("____________________________________");
System.out.println();
System.out.println("Whole : " + mS1R + " ms");
System.out.println();
System.out.println("1/2 : " + mS2R + " ms");
for (int x = 1; x < 9; x = x * 2) { // 遵循命名规范,x 小写
double mS = Math.round((60000.00 / x) / bpm);
int y = x * 4; // 遵循命名规范,y 小写
int mSR = (int) mS;
System.out.println();
System.out.println("1/" + y + " : " + mSR + " ms");
}
System.out.println("____________________________________");
System.out.println();
}
public static int getUserInt(Scanner scanner, int min, int max) { // 接收 Scanner 作为参数
int userInput = min - 1;
boolean keepAsking = true;
while (keepAsking || userInput < min || userInput > max) {
try {
userInput = scanner.nextInt();
if (userInput >= min && userInput <= max) {
keepAsking = false;
} else {
System.out.println("输入超出范围,请在 " + min + " 到 " + max + " 之间输入。");
}
} catch (InputMismatchException e) { // 捕获更具体的异常
System.out.println("无效输入,请输入一个整数。");
} finally {
scanner.nextLine(); // 消费掉无效输入或回车符
}
}
System.out.println(userInput);
return userInput;
}
}在这个重构的示例中,Scanner对象在main方法中创建一次,并通过参数传递给需要它的方法。程序通过while循环来控制是否重复执行计算逻辑,而不是通过递归调用main。
正确处理Scanner(System.in)的关闭问题是Java编程中的一个重要细节。核心原则是:不要关闭System.in,因为它不是你创建的资源。 对于其他由你创建并需要关闭的资源,应优先使用try-with-resources语句来确保资源的自动释放。同时,遵循Java的命名规范,并避免不当的递归调用,将有助于编写出更健壮、可读性更强的代码。理解这些基础概念和最佳实践,是成为一名优秀Java开发者的必经之路。
以上就是Java中Scanner与System.in的正确关闭姿势与常见误区解析的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号