该Java教程是针对JDK 8编写的。本页面描述的示例和实践不利用后续版本引入的改进,并且可能使用不再可用的技术。
请参阅Java语言变更,了解Java SE 9及其后续版本中更新的语言功能的摘要。
请参阅JDK发行说明,了解有关所有JDK版本的新功能、增强功能和已删除或弃用选项的信息。
程序使用字节流来执行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包含字符数据,最好的方法是使用字符流,如下一节所讨论的。还有更复杂数据类型的流。字节流只应用于最基本的I/O。
那么为什么要讨论字节流呢?因为所有其他流类型都是基于字节流构建的。