在Java中如何处理输入输出异常_Java基础语法注意事项解析

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

Java中Scanner读取输入时抛出InputMismatchException怎么办

这是新手最常遇到的输入异常之一:用户本该输数字,却输了字母,Scanner在调用nextInt()nextDouble()时直接炸掉,程序中断。根本原因不是“没处理异常”,而是Scanner在匹配失败后**不消费输入流中的非法字符**,导致下次读取仍卡在同一位置。

实操建议:

  • 改用nextLine()统一读取字符串,再用Integer.parseInt()等转换,并包在try-catch里捕获NumberFormatException
  • 若坚持用nextInt()类方法,捕获InputMismatchException后必须调用scanner.next()(不是nextLine())清掉错误token
  • 避免混用nextLine()nextXXX()——前者会吃掉换行符,后者不会,容易造成“跳过输入”的假象

FileInputStream读文件时抛FileNotFoundException的常见误判

这个异常不是“文件真不存在”才触发,而是JVM找不到路径所指的资源。多数情况是路径写错了,但错得很有迷惑性。

实操建议:

  • "data.txt"是相对路径,基准目录是**当前工作目录(current working directory)**,不是源码所在目录,更不是src目录——IDE运行时通常以项目根目录为基准
  • getClass().getResource("/data.txt")获取类路径下的资源,返回URL,再转FileInputStream,适合配置文件、模板等打包进jar的场景
  • 绝对路径慎用:"C:\config\test.txt"在Windows下要写成"C:\\config\\test.txt""C:/config/test.txt",否则反斜杠被当转义符

try-with-resources为什么不能替代finally里的close()

能替代——但仅限于实现了AutoCloseable接口的对象(如FileInputStreamBufferedReader)。很多人以为只要写了try-with-resources就万事大吉,结果还是遇到IOException没处理,或者关闭时抛新异常把原异常吞掉。

实操建议:

  • 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默认字符集,就乱码。

实操建议:

  • 查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 (!scann

er.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,都得明确知道它背后连着哪条链路。