InputMismatchException需调用scanner.next()清除非法输入;FileNotFoundException因路径基准是工作目录而非源码目录;try-with-resources自动关闭AutoCloseable资源但可能抑制异常;控制台中文乱码源于终端与JVM编码不一致。

Java中Scanner读取输入时抛出InputMismatchException怎么办
这是新手最常遇到的输入异常之一:用户本该输数字,却输了字母,Scanner在调用nextInt()或nextDouble()时直接炸掉,程序中断。根本原因不是“没处理异常”,而是Scanner在匹配失败后**不消费输入流中的非法字符**,导致下次读取仍卡在同一位置。
实操建议:
立即学习“Java免费学习笔记(深入)”;
- 改用
nextLine()统一读取字符串,再用Integer.parseInt()等转换,并包在try-catch里捕获NumberFormatException - 若坚持用
nextInt()类方法,捕获InputMismatchException后必须调用scanner.next()(不是nextLine())清掉错误token - 避免混用
nextLine()和nextXXX()——前者会吃掉换行符,后者不会,容易造成“跳过输入”的假象
用FileInputStream读文件时抛FileNotFoundException的常见误判
这个异常不是“文件真不存在”才触发,而是JVM找不到路径所指的资源。多数情况是路径写错了,但错得很有迷惑性。
实操建议:
立即学习“Java免费学习笔记(深入)”;
-
"data.txt"是相对路径,基准目录是**当前工作目录(current working directory)**,不是源码所在目录,更不是src目录——IDE运行时通常以项目根目录为基准 - 用
getClass().getResource("/data.txt")获取类路径下的资源,返回URL,再转File或InputStream,适合配置文件、模板等打包进jar的场景 - 绝对路径慎用:
"C:\config\test.txt"在Windows下要写成"C:\\config\\test.txt"或"C:/config/test.txt",否则反斜杠被当转义符
try-with-resources为什么不能替代finally里的close()
能替代——但仅限于实现了AutoCloseable接口的对象(如FileInputStream、BufferedReader)。很多人以为只要写了try-with-resources就万事大吉,结果还是遇到IOException没处理,或者关闭时抛新异常把原异常吞掉。
实操建议:
立即学习“Java免费学习笔记(深入)”;
-
try-with-resources语句中声明的资源,会在块结束时自动调用close(),但close()本身可能抛IOException,它会被抑制(suppressed),需通过Throwable.getSuppressed()获取 - 如果需要在关闭失败时做特殊处理(比如记录日志、重试),就不能依赖自动关闭,得手动写
finally块 - 多个资源用分号隔开:
try (FileInputStream fis = new FileInputStream("a"); BufferedReader br = new BufferedReader(new InputStreamReader(fis))) { ... },注意顺序:后声明的先关闭
控制台输出中文乱码(System.out.println("你好")显示问号)
本质是编码不一致:Java字符串内部用UTF-16,但终端/命令行默认用系统本地编码(Windows是GBK,Linux/macOS通常是UTF-8)。当终端编码≠JVM默认字符集,就乱码。
实操建议:
立即学习“Java免费学习笔记(深入)”;
- 查JVM默认编码:
System.getProperty("file.encoding"),在启动时加-Dfile.encoding=UTF-8强制统一 - IDE运行时,在运行配置里设置VM options(IntelliJ:Run → Edit Configurations → VM options)
- 命令行运行时,Windows下先执行
chcp 65001切换到UTF-8页,再java -Dfile.encoding=UTF-8 MyApp - 不推荐用
new PrintStream(System.out, true, "UTF-8")包装输出流——治标不治本,且影响其他输出
public class IoDemo {
public static void main(String[] args) {
// ✅ 安全读整数
Scanner scanner = new Scanner(System.in);
System.out.print("请输入一个整数:");
while (!scanner.hasNextInt()) {
System.out.print("输入无效,请重新输入整数:");
scanner.next(); // 清掉非法输入
}
int n = scanner.nextInt();
System.out.println("你输入的是:" + n);
// ✅ 安全读文件(类路径资源)
try (InputStream is = IoDemo.class.getResourceAsStream("/config.json")) {
if (is == null) {
throw new RuntimeException("资源未找到:/config.json");
}
// 处理输入流...
} catch (IOException e) {
e.printStackTrace();
}
}
}
路径、编码、资源生命周期这三处,任何一个细节偏差都会让IO操作当场失效。别信“应该没问题”,每个new FileInputStream、每行System.out.println,都得明确知道它背后连着哪条链路。










