Java 教程是为 JDK 8 编写的。本页中描述的示例和实践未利用在后续版本中引入的改进。
程序使用 byte streams (字节流) 执行 8 位字节的输入和输出。所有字节流类都继承自 InputStream
和 OutputStream
。
有许多字节流类。为了演示字节流的工作原理,我们将重点关注文件 I/O 字节流,FileInputStream
和 FileOutputStream
。其他种类的字节流的使用方式大致相同;它们的不同之处主要在于它们的构造方式。
我们将通过检查名为 CopyBytes
的示例程序来探索 FileInputStream
和 FileOutputStream
,该程序使用字节流来复制 xanadu.txt
,一次一个字节。
import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOException; public class CopyBytes { public static void main(String[] args) throws IOException { FileInputStream in = null; FileOutputStream out = null; try { in = new FileInputStream("xanadu.txt"); out = new FileOutputStream("outagain.txt"); int c; while ((c = in.read()) != -1) { out.write(c); } } finally { if (in != null) { in.close(); } if (out != null) { out.close(); } } } }
CopyBytes
将大部分时间花在一个简单的循环中,该循环读取输入流并一次写入一个字节的输出流,如 下图 所示。
简单的字节流输入和输出。
在不再需要流时关闭流是非常重要的 非常重要以至于 CopyBytes
使用 finally
块来保证即使发生错误也会关闭两个流。这种做法有助于避免严重的资源泄漏。
一个可能的错误是 CopyBytes
无法打开一个或两个文件。发生这种情况时,对应于该文件的流变量永远不会更改其初始值(null
)。这就是 CopyBytes
确保每个流变量在调用 close
之前包含对象引用的原因。
CopyBytes
似乎是一个普通的程序,但它实际上代表了一种你应该避免的低级 I/O 。由于 xanadu.txt
包含字符数据,因此最好的方法是使用 character streams,如下一节所述。还有用于更复杂数据类型的流。字节流只能用于最基本的 I/O。
那么,为什么要谈论字节流呢?因为所有其他流类型都是基于字节流构建的。