登录
首页 >  文章 >  java教程

Java输入输出类库与FileInputStream详解

时间:2026-03-02 13:32:41 255浏览 收藏

本文深入剖析了Java中FileInputStream使用中的常见陷阱与最佳实践,揭示了FileNotFoundException频发的根源在于路径解析基准是当前工作目录而非类路径,并强调需通过System.getProperty("user.dir")确认实际路径、对classpath资源改用getResourceAsStream();针对性能问题,指出BufferedInputStream对中大文件读取的必要性,但必须配合read(byte[])批量读而非单字节调用;同时警示FileInputStream与Scanner混用会导致数据丢失或阻塞,推荐统一使用高层API并显式指定编码;最后提醒available()方法不可靠,避免误用。这些实战经验直击开发者日常踩坑痛点,助你写出更健壮、高效、可维护的IO代码。

Java常用输入输出类库与FileInputStream

FileInputStream 读取文件时为什么总是抛出 FileNotFoundException?

根本原因通常是路径没写对,而不是文件真丢了。Java 的 FileInputStream 默认以当前工作目录(不是类路径、也不是 src 目录)为基准解析相对路径。

  • 运行 JAR 包时,"data.txt" 指的是 JAR 所在目录下的 data.txt,不是 JAR 包内部的资源
  • IDE 运行时,当前工作目录通常是项目根目录,但不同 IDE(IntelliJ / Eclipse)可能有差异,建议用 System.getProperty("user.dir") 打印确认
  • 若文件在 classpath 下(如 src/main/resources/config.json),别用 FileInputStream,改用 Class.getResourceAsStream()

BufferedInputStream + FileInputStream 组合还有必要吗?

有必要,尤其读取中大文件(几 MB 以上)时。直接用 FileInputStream.read() 单字节读取,系统调用太频繁,性能极差;而 BufferedInputStream 自动缓存,默认 8192 字节缓冲区,能大幅减少底层 I/O 次数。

  • 不要手动包装成 new BufferedInputStream(new FileInputStream(...)) 后再调用 read() 单字节——这仍慢,应改用 read(byte[]) 批量读
  • JDK 9+ 推荐优先用 Files.readAllBytes(Paths.get(...)) 读小文件(
  • 注意:BufferedInputStream 不改变异常类型,FileNotFoundExceptionIOException 仍需显式捕获或声明

FileInputStream 与 Scanner 混用会出什么问题?

会丢数据,甚至阻塞。因为 Scanner 内部也维护缓冲区,它可能从 FileInputStream 预读若干字节(比如为了判断下一行是否存在),而你后续又用原流的 read(),就会跳过这部分已读未消费的数据。

  • 选一个:要么全程用 Scanner(适合按行/按词解析文本),要么全程用 FileInputStream + InputStreamReader + BufferedReader(更可控、支持编码指定)
  • Scanner 默认使用平台默认编码,中文环境易乱码;务必用 new Scanner(inputStream, "UTF-8") 显式指定
  • 关闭流时,只关外层(如 ScannerBufferedReader),它会自动委托关闭底层 FileInputStream
try (FileInputStream fis = new FileInputStream("log.bin");
     BufferedInputStream bis = new BufferedInputStream(fis)) {
    byte[] buf = new byte[4096];
    int len;
    while ((len = bis.read(buf)) != -1) {
        // 处理 buf[0] 到 buf[len-1]
    }
} catch (IOException e) {
    // 注意:FileInputStream 构造本身可能抛出 FileNotFoundException
    // 它是 IOException 子类,统一 catch 即可
}
真正容易被忽略的是:FileInputStreamavailable() 方法返回值不可靠,不能用来判断是否读完或分配缓冲区大小;它只是“当前可不阻塞读取的估计字节数”,对文件可能返回全部长度,也可能只返回部分,取决于底层实现和 OS。

文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《Java输入输出类库与FileInputStream详解》文章吧,也可关注golang学习网公众号了解相关技术文章。

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>