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。
那么,为什么要谈论字节流呢?因为所有其他流类型都是基于字节流构建的。